blob: 259386c79b1aad3f97a97d5844b5005d46002f7c [file] [log] [blame]
// Copyright 2021 Google LLC
//
// Licensed under the Apache License, Version 2.0 (the "License");
// you may not use this file except in compliance with the License.
// You may obtain a copy of the License at
//
// http://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
// See the License for the specific language governing permissions and
// limitations under the License.
#include "ncsi_sockio.h"
#include "common_defs.h"
#include "net_iface.h"
#include <linux/filter.h>
#include <linux/if.h>
#include <linux/if_ether.h>
#include <netinet/in.h>
#include <poll.h>
#include <sys/socket.h>
#include <cstring>
#include <iterator>
namespace ncsi
{
int SockIO::init()
{
RETURN_IF_ERROR(sockfd_ = socket(AF_PACKET, SOCK_RAW, htons(ETH_P_ALL)),
"ncsi::SockIO::init() failed");
return 0;
}
int SockIO::bind_to_iface(const net::IFaceBase& iface)
{
iface.set_sock_flags(sockfd_, IFF_PROMISC);
std::memset(&sock_addr_, 0, sizeof(sock_addr_));
sock_addr_.sll_family = AF_PACKET;
sock_addr_.sll_protocol = htons(ETH_P_ALL);
RETURN_IF_ERROR(iface.bind_sock(sockfd_, &sock_addr_),
"ncsi::SockIO::bind_to_iface failed");
return 0;
}
/**
* Drops VLAN tagged packets from a socket
*
* ld vlant
* jneq #0, drop
* ld proto
* jneq #0x88f8, drop
* ret #-1
* drop: ret #0
*/
struct sock_filter vlan_remove_code[] = {
{0x20, 0, 0, 0xfffff02c}, {0x15, 0, 3, 0x00000000},
{0x20, 0, 0, 0xfffff000}, {0x15, 0, 1, 0x000088f8},
{0x6, 0, 0, 0xffffffff}, {0x6, 0, 0, 0x00000000}};
struct sock_fprog vlan_remove_bpf = {
std::size(vlan_remove_code),
vlan_remove_code,
};
int SockIO::filter_vlans()
{
return setsockopt(sockfd_, SOL_SOCKET, SO_ATTACH_FILTER, &vlan_remove_bpf,
sizeof(vlan_remove_bpf));
}
int SockIO::recv(void* buf, size_t maxlen)
{
struct pollfd sock_pollfd
{
sockfd_, POLLIN | POLLPRI, 0
};
int ret = poll(&sock_pollfd, 1, kpoll_timeout_);
if (ret > 0)
{
return ::recv(sockfd_, buf, maxlen, 0);
}
return ret;
}
} // namespace ncsi