157 lines
		
	
	
		
			2.9 KiB
		
	
	
	
		
			C
		
	
	
	
			
		
		
	
	
			157 lines
		
	
	
		
			2.9 KiB
		
	
	
	
		
			C
		
	
	
	
| // SPDX-License-Identifier: GPL-2.0
 | |
| /*
 | |
|  * Copyright 2019 Ramon Fried <rfried.dev@gmail.com>
 | |
|  */
 | |
| 
 | |
| #include <common.h>
 | |
| #include <net.h>
 | |
| #include <net/pcap.h>
 | |
| #include <time.h>
 | |
| #include <asm/io.h>
 | |
| 
 | |
| #define LINKTYPE_ETHERNET	1
 | |
| 
 | |
| static bool initialized;
 | |
| static bool running;
 | |
| static bool buffer_full;
 | |
| static void *buf;
 | |
| static unsigned int max_size;
 | |
| static unsigned int pos;
 | |
| 
 | |
| static unsigned long incoming_count;
 | |
| static unsigned long outgoing_count;
 | |
| 
 | |
| struct pcap_header {
 | |
| 	u32 magic;
 | |
| 	u16 version_major;
 | |
| 	u16 version_minor;
 | |
| 	s32 thiszone;
 | |
| 	u32 sigfigs;
 | |
| 	u32 snaplen;
 | |
| 	u32 network;
 | |
| };
 | |
| 
 | |
| struct pcap_packet_header {
 | |
| 	u32 ts_sec;
 | |
| 	u32 ts_usec;
 | |
| 	u32 incl_len;
 | |
| 	u32 orig_len;
 | |
| };
 | |
| 
 | |
| static struct pcap_header file_header = {
 | |
| 	.magic = 0xa1b2c3d4,
 | |
| 	.version_major = 2,
 | |
| 	.version_minor = 4,
 | |
| 	.snaplen = 65535,
 | |
| 	.network = LINKTYPE_ETHERNET,
 | |
| };
 | |
| 
 | |
| int pcap_init(phys_addr_t paddr, unsigned long size)
 | |
| {
 | |
| 	buf = map_physmem(paddr, size, 0);
 | |
| 	if (!buf) {
 | |
| 		printf("Failed mapping PCAP memory\n");
 | |
| 		return -ENOMEM;
 | |
| 	}
 | |
| 
 | |
| 	printf("PCAP capture initialized: addr: 0x%lx max length: %lu\n",
 | |
| 	       (unsigned long)buf, size);
 | |
| 
 | |
| 	memcpy(buf, &file_header, sizeof(file_header));
 | |
| 	pos = sizeof(file_header);
 | |
| 	max_size = size;
 | |
| 	initialized = true;
 | |
| 	running = false;
 | |
| 	buffer_full = false;
 | |
| 	incoming_count = 0;
 | |
| 	outgoing_count = 0;
 | |
| 	return 0;
 | |
| }
 | |
| 
 | |
| int pcap_start_stop(bool start)
 | |
| {
 | |
| 	if (!initialized) {
 | |
| 		printf("error: pcap was not initialized\n");
 | |
| 		return -ENODEV;
 | |
| 	}
 | |
| 
 | |
| 	running = start;
 | |
| 
 | |
| 	return 0;
 | |
| }
 | |
| 
 | |
| int pcap_clear(void)
 | |
| {
 | |
| 	if (!initialized) {
 | |
| 		printf("error: pcap was not initialized\n");
 | |
| 		return -ENODEV;
 | |
| 	}
 | |
| 
 | |
| 	pos = sizeof(file_header);
 | |
| 	incoming_count = 0;
 | |
| 	outgoing_count = 0;
 | |
| 	buffer_full = false;
 | |
| 
 | |
| 	printf("pcap capture cleared\n");
 | |
| 	return 0;
 | |
| }
 | |
| 
 | |
| int pcap_post(const void *packet, size_t len, bool outgoing)
 | |
| {
 | |
| 	struct pcap_packet_header header;
 | |
| 	u64 cur_time = timer_get_us();
 | |
| 
 | |
| 	if (!initialized || !running || !buf)
 | |
| 		return -ENODEV;
 | |
| 
 | |
| 	if (buffer_full)
 | |
| 		return -ENOMEM;
 | |
| 
 | |
| 	if ((pos + len + sizeof(header)) >= max_size) {
 | |
| 		buffer_full = true;
 | |
| 		printf("\n!!! Buffer is full, consider increasing buffer size !!!\n");
 | |
| 		return -ENOMEM;
 | |
| 	}
 | |
| 
 | |
| 	header.ts_sec = cur_time / 1000000;
 | |
| 	header.ts_usec = cur_time % 1000000;
 | |
| 	header.incl_len = len;
 | |
| 	header.orig_len = len;
 | |
| 
 | |
| 	memcpy(buf + pos, &header, sizeof(header));
 | |
| 	pos += sizeof(header);
 | |
| 	memcpy(buf + pos, packet, len);
 | |
| 	pos += len;
 | |
| 
 | |
| 	if (outgoing)
 | |
| 		outgoing_count++;
 | |
| 	else
 | |
| 		incoming_count++;
 | |
| 
 | |
| 	env_set_hex("pcapsize", pos);
 | |
| 
 | |
| 	return 0;
 | |
| }
 | |
| 
 | |
| int pcap_print_status(void)
 | |
| {
 | |
| 	if (!initialized) {
 | |
| 		printf("pcap was not initialized\n");
 | |
| 		return -ENODEV;
 | |
| 	}
 | |
| 	printf("PCAP status:\n");
 | |
| 	printf("\tInitialized addr: 0x%lx\tmax length: %u\n",
 | |
| 	       (unsigned long)buf, max_size);
 | |
| 	printf("\tStatus: %s.\t file size: %u\n", running ? "Active" : "Idle",
 | |
| 	       pos);
 | |
| 	printf("\tIncoming packets: %lu Outgoing packets: %lu\n",
 | |
| 	       incoming_count, outgoing_count);
 | |
| 
 | |
| 	return 0;
 | |
| }
 | |
| 
 | |
| bool pcap_active(void)
 | |
| {
 | |
| 	return running;
 | |
| }
 |