11285115cSWilliam A. Kennington III // Copyright 2021 Google LLC 21285115cSWilliam A. Kennington III // 31285115cSWilliam A. Kennington III // Licensed under the Apache License, Version 2.0 (the "License"); 41285115cSWilliam A. Kennington III // you may not use this file except in compliance with the License. 51285115cSWilliam A. Kennington III // You may obtain a copy of the License at 61285115cSWilliam A. Kennington III // 71285115cSWilliam A. Kennington III // http://www.apache.org/licenses/LICENSE-2.0 81285115cSWilliam A. Kennington III // 91285115cSWilliam A. Kennington III // Unless required by applicable law or agreed to in writing, software 101285115cSWilliam A. Kennington III // distributed under the License is distributed on an "AS IS" BASIS, 111285115cSWilliam A. Kennington III // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 121285115cSWilliam A. Kennington III // See the License for the specific language governing permissions and 131285115cSWilliam A. Kennington III // limitations under the License. 141285115cSWilliam A. Kennington III 151285115cSWilliam A. Kennington III #include "ncsi_sockio.h" 161285115cSWilliam A. Kennington III 171285115cSWilliam A. Kennington III #include "common_defs.h" 181285115cSWilliam A. Kennington III #include "net_iface.h" 191285115cSWilliam A. Kennington III 201285115cSWilliam A. Kennington III #include <linux/filter.h> 21adb8ffe4SWilly Tu #include <linux/if.h> 22adb8ffe4SWilly Tu #include <linux/if_ether.h> 231285115cSWilliam A. Kennington III #include <netinet/in.h> 241285115cSWilliam A. Kennington III #include <poll.h> 251285115cSWilliam A. Kennington III #include <sys/socket.h> 261285115cSWilliam A. Kennington III 271285115cSWilliam A. Kennington III #include <cstring> 28adb8ffe4SWilly Tu #include <iterator> 291285115cSWilliam A. Kennington III 301285115cSWilliam A. Kennington III namespace ncsi 311285115cSWilliam A. Kennington III { 321285115cSWilliam A. Kennington III 331285115cSWilliam A. Kennington III int SockIO::init() 341285115cSWilliam A. Kennington III { 351285115cSWilliam A. Kennington III RETURN_IF_ERROR(sockfd_ = socket(AF_PACKET, SOCK_RAW, htons(ETH_P_ALL)), 361285115cSWilliam A. Kennington III "ncsi::SockIO::init() failed"); 371285115cSWilliam A. Kennington III return 0; 381285115cSWilliam A. Kennington III } 391285115cSWilliam A. Kennington III 401285115cSWilliam A. Kennington III int SockIO::bind_to_iface(const net::IFaceBase& iface) 411285115cSWilliam A. Kennington III { 421285115cSWilliam A. Kennington III iface.set_sock_flags(sockfd_, IFF_PROMISC); 431285115cSWilliam A. Kennington III 44a9a98258SWilly Tu RETURN_IF_ERROR(iface.bind_sock(sockfd_), 451285115cSWilliam A. Kennington III "ncsi::SockIO::bind_to_iface failed"); 461285115cSWilliam A. Kennington III 471285115cSWilliam A. Kennington III return 0; 481285115cSWilliam A. Kennington III } 491285115cSWilliam A. Kennington III 501285115cSWilliam A. Kennington III /** 511285115cSWilliam A. Kennington III * Drops VLAN tagged packets from a socket 521285115cSWilliam A. Kennington III * 531285115cSWilliam A. Kennington III * ld vlant 541285115cSWilliam A. Kennington III * jneq #0, drop 551285115cSWilliam A. Kennington III * ld proto 561285115cSWilliam A. Kennington III * jneq #0x88f8, drop 571285115cSWilliam A. Kennington III * ret #-1 581285115cSWilliam A. Kennington III * drop: ret #0 591285115cSWilliam A. Kennington III */ 601285115cSWilliam A. Kennington III struct sock_filter vlan_remove_code[] = { 611285115cSWilliam A. Kennington III {0x20, 0, 0, 0xfffff02c}, {0x15, 0, 3, 0x00000000}, 621285115cSWilliam A. Kennington III {0x20, 0, 0, 0xfffff000}, {0x15, 0, 1, 0x000088f8}, 631285115cSWilliam A. Kennington III {0x6, 0, 0, 0xffffffff}, {0x6, 0, 0, 0x00000000}}; 641285115cSWilliam A. Kennington III 651285115cSWilliam A. Kennington III struct sock_fprog vlan_remove_bpf = { 661285115cSWilliam A. Kennington III std::size(vlan_remove_code), 671285115cSWilliam A. Kennington III vlan_remove_code, 681285115cSWilliam A. Kennington III }; 691285115cSWilliam A. Kennington III 701285115cSWilliam A. Kennington III int SockIO::filter_vlans() 711285115cSWilliam A. Kennington III { 721285115cSWilliam A. Kennington III return setsockopt(sockfd_, SOL_SOCKET, SO_ATTACH_FILTER, &vlan_remove_bpf, 731285115cSWilliam A. Kennington III sizeof(vlan_remove_bpf)); 741285115cSWilliam A. Kennington III } 751285115cSWilliam A. Kennington III 761285115cSWilliam A. Kennington III int SockIO::recv(void* buf, size_t maxlen) 771285115cSWilliam A. Kennington III { 78*d980e4fcSPatrick Williams struct pollfd sock_pollfd{sockfd_, POLLIN | POLLPRI, 0}; 791285115cSWilliam A. Kennington III 801285115cSWilliam A. Kennington III int ret = poll(&sock_pollfd, 1, kpoll_timeout_); 811285115cSWilliam A. Kennington III if (ret > 0) 821285115cSWilliam A. Kennington III { 831285115cSWilliam A. Kennington III return ::recv(sockfd_, buf, maxlen, 0); 841285115cSWilliam A. Kennington III } 851285115cSWilliam A. Kennington III 861285115cSWilliam A. Kennington III return ret; 871285115cSWilliam A. Kennington III } 881285115cSWilliam A. Kennington III 891285115cSWilliam A. Kennington III } // namespace ncsi 90