diff options
38 files changed, 1217 insertions, 1275 deletions
| diff --git a/host/apps/omap_debug/.gitignore b/host/apps/omap_debug/.gitignore deleted file mode 100644 index 008a23138..000000000 --- a/host/apps/omap_debug/.gitignore +++ /dev/null @@ -1,20 +0,0 @@ -.gitignore -clkgen-config -fpga-downloader -usrp-e-button -usrp-e-crc-rw -usrp-e-ctl -usrp-e-debug-pins -usrp-e-fpga-rw -usrp-e-gpio -usrp-e-i2c -usrp-e-lb-test -usrp-e-led -usrp-e-loopback -usrp-e-random-loopback -usrp-e-rw -usrp-e-spi -usrp-e-timed -usrp-e-uart -usrp-e-uart-rx -usrp-e-mm-loopback diff --git a/host/apps/omap_debug/Makefile b/host/apps/omap_debug/Makefile deleted file mode 100644 index f8b9f2bd9..000000000 --- a/host/apps/omap_debug/Makefile +++ /dev/null @@ -1,33 +0,0 @@ -CFLAGS=-Wall -I../../lib/usrp/usrp_e/ -march=armv7-a -mtune=cortex-a8 -mfpu=neon -O3 -CXXFLAGS=-Wall -I../../lib/usrp/usrp_e/ -march=armv7-a -mtune=cortex-a8 -mfpu=neon -O3 - -all : usrp-e-spi usrp-e-i2c usrp-e-uart usrp-e-led usrp-e-ctl usrp-e-button usrp-e-uart-rx usrp-e-gpio usrp-e-debug-pins - -usrp-e-spi : usrp-e-spi.c - -usrp-e-i2c : usrp-e-i2c.c - -usrp-e-uart : usrp-e-uart.c - -usrp-e-uart-rx : usrp-e-uart-rx.c - -usrp-e-led : usrp-e-led.c - -usrp-e-ctl : usrp-e-ctl.c - -usrp-e-button : usrp-e-button.c - -usrp-e-gpio : usrp-e-gpio.c - -usrp-e-debug-pins : usrp-e-debug-pins.c -clean : -	rm -f usrp-e-spi -	rm -f usrp-e-i2c -	rm -f usrp-e-uart -	rm -f usrp-e-uart-rx -	rm -f usrp-e-led -	rm -f usrp-e-ctl -	rm -f usrp-e-button -	rm -f usrp-e-gpio -	rm -f usrp-e-debug-pins -	rm -f usrp-e-lb-test diff --git a/host/apps/omap_debug/set_debug_pins.py b/host/apps/omap_debug/set_debug_pins.py deleted file mode 100755 index 0f9ecd7b9..000000000 --- a/host/apps/omap_debug/set_debug_pins.py +++ /dev/null @@ -1,35 +0,0 @@ -#!/usr/bin/python - -import os - -# Memory Map -misc_base = 0 -uart_base = 1 -spi_base = 2 -i2c_base = 3 -gpio_base = 4 * 128 -settings_base = 5 - -# GPIO offset -gpio_pins = 0 -gpio_ddr = 4 -gpio_ctrl_lo = 8 -gpio_ctrl_hi = 12 - -def set_reg(reg, val): -    os.system("./usrp1-e-ctl w %d 1 %d" % (reg,val)) - -def get_reg(reg): -    fin,fout = os.popen4("./usrp1-e-ctl r %d 1" % (reg,)) -    print fout.read() - -# Set DDRs to output -set_reg(gpio_base+gpio_ddr, 0xFFFF) -set_reg(gpio_base+gpio_ddr+2, 0xFFFF) - -# Set CTRL to Debug #0 ( A is for debug 0, F is for debug 1 ) -set_reg(gpio_base+gpio_ctrl_lo, 0xAAAA) -set_reg(gpio_base+gpio_ctrl_lo+2, 0xAAAA) -set_reg(gpio_base+gpio_ctrl_hi, 0xAAAA) -set_reg(gpio_base+gpio_ctrl_hi+2, 0xAAAA) - diff --git a/host/apps/omap_debug/test.c b/host/apps/omap_debug/test.c deleted file mode 100644 index 36f4d700a..000000000 --- a/host/apps/omap_debug/test.c +++ /dev/null @@ -1,34 +0,0 @@ -#include <stdio.h> - -void -main() -{ -  int x; -  char *y; -  long long z; - -  x = 0x01020304; -  z = 0x0102030405060708LL; - -  printf("%x\n",x); -  y = (char *)&x; -  printf("%x\n",y[0]); -  printf("%x\n",y[1]); -  printf("%x\n",y[2]); -  printf("%x\n",y[3]); - -  printf("Printing z ...\n"); -  printf("%llx\n",z); -  printf("Printing z done\n"); - -  y = (char *)&z; -  printf("%x\n",y[0]); -  printf("%x\n",y[1]); -  printf("%x\n",y[2]); -  printf("%x\n",y[3]); -  printf("%x\n",y[4]); -  printf("%x\n",y[5]); -  printf("%x\n",y[6]); -  printf("%x\n",y[7]); -} - diff --git a/host/apps/omap_debug/u1e-read-stream.c b/host/apps/omap_debug/u1e-read-stream.c deleted file mode 100644 index 4e4c21d9e..000000000 --- a/host/apps/omap_debug/u1e-read-stream.c +++ /dev/null @@ -1,21 +0,0 @@ -#include <stdio.h> -#include <sys/types.h> -#include <fcntl.h> - -int main(int rgc, char *argv[]) -{ -	int fp, cnt, n; -	short buf[1024]; - -	n = 0; - -	fp = open("/dev/usrp1_e0", O_RDONLY); -	printf("fp = %d\n", fp); - -	do { -		cnt = read(fp, buf, 2048); -		n++; -//		printf("Bytes read - %d\n", cnt); -	} while(n < 10*512); -	printf("Data - %hX\n", buf[0]); -} diff --git a/host/apps/omap_debug/usrp-e-button.c b/host/apps/omap_debug/usrp-e-button.c deleted file mode 100644 index f13291491..000000000 --- a/host/apps/omap_debug/usrp-e-button.c +++ /dev/null @@ -1,56 +0,0 @@ -#include <stdio.h> -#include <stdlib.h> -#include <sys/types.h> -#include <fcntl.h> -#include <string.h> -#include <sys/ioctl.h> -#include <unistd.h> - -#include "usrp_e.h" -#include "usrp_e_regs.hpp" - -// Usage: usrp_e_uart <string> - -#define PB1 (1<<8) -#define PB2 (1<<9) -#define PB3 (1<<10) -#define P1 (0) -#define P2 (0xFF) -#define P3 (0xAA) -#define P4 (0x55) - -int main(int argc, char *argv[]) -{ -	int fp, ret; -	struct usrp_e_ctl16 d; -	int pb1=0, pb2=0, pb3=0, p1=0, p2=0, p3=0, p4=0; - -	fp = open("/dev/usrp_e0", O_RDWR); -	printf("fp = %d\n", fp); - -	d.offset = UE_REG_MISC_SW; -	d.count = 1; - -	do { -		ret = ioctl(fp, USRP_E_READ_CTL16, &d); -		if (d.buf[0] & PB1) { -			pb1 = 1; -			printf("Pushbutton 1 hit\n"); -		} - -		if (d.buf[0] & PB2) { -			pb2 = 1; -			printf("Pushbutton 2 hit\n"); -		} - -		if (d.buf[0] & PB3) { -			pb3 = 1; -			printf("Pushbutton 3 hit\n"); -		} - -		sleep(1); - -	} while (!(pb1 && pb2 && pb3)); - -	return 0; -} diff --git a/host/apps/omap_debug/usrp-e-ctl.c b/host/apps/omap_debug/usrp-e-ctl.c deleted file mode 100644 index 69c48ee6f..000000000 --- a/host/apps/omap_debug/usrp-e-ctl.c +++ /dev/null @@ -1,48 +0,0 @@ -#include <stdio.h> -#include <stdlib.h> -#include <sys/types.h> -#include <fcntl.h> -#include <sys/ioctl.h> - -#include "usrp_e.h" - -// Usage: usrp_e_ctl w|r offset number_of_values val1 val2 .... - -int main(int argc, char *argv[]) -{ -	int fp, i, cnt, ret; -	struct usrp_e_ctl16 ctl_data; - -	if (argc < 4) { -		printf("Usage: usrp_e_ctl w|r offset number_of_values val1 val2 ....\n"); -		exit(-1); -	} - -	cnt = atoi(argv[3]); - -	ctl_data.offset = atoi(argv[2]); -	ctl_data.count  = cnt; - -	fp = open("/dev/usrp_e0", O_RDWR); -	printf("fp = %d\n", fp); - -	if (*argv[1] == 'w') { -		for (i=0; i<cnt; i++) -			ctl_data.buf[i] = atoi(argv[4+i]); - -		ret = ioctl(fp, USRP_E_WRITE_CTL16, &ctl_data); -		printf("Return value from write ioctl = %d\n", ret); -	} - -	if (*argv[1] == 'r') { -		ret = ioctl(fp, USRP_E_READ_CTL16, &ctl_data); -		printf("Return value from write ioctl = %d\n", ret); - -		for (i=0; i<ctl_data.count; i++) { -			if (!(i%8)) -				printf("\nData at %4d :", i); -			printf(" %5d", ctl_data.buf[i]); -		} -		printf("\n"); -	} -} diff --git a/host/apps/omap_debug/usrp-e-debug-pins.c b/host/apps/omap_debug/usrp-e-debug-pins.c deleted file mode 100644 index d18bbf990..000000000 --- a/host/apps/omap_debug/usrp-e-debug-pins.c +++ /dev/null @@ -1,77 +0,0 @@ -#include <stdio.h> -#include <stdlib.h> -#include <unistd.h> -#include <sys/types.h> -#include <fcntl.h> -#include <string.h> -#include <sys/ioctl.h> - -#include "usrp_e.h" -#include "usrp_e_regs.hpp" - -// Usage: usrp_e_gpio <string> - -static int fp; - -static int read_reg(__u16 reg) -{ -	int ret; -	struct usrp_e_ctl16 d; - -	d.offset = reg; -	d.count = 1; -	ret = ioctl(fp, USRP_E_READ_CTL16, &d); -	return d.buf[0]; -} - -static void write_reg(__u16 reg, __u16 val) -{ -	int ret; -	struct usrp_e_ctl16 d; - -	d.offset = reg; -	d.count = 1; -	d.buf[0] = val; -	ret = ioctl(fp, USRP_E_WRITE_CTL16, &d); -} - -int main(int argc, char *argv[]) -{ -	int test; - -	test = 0; -	if (argc < 2) { -		printf("%s 0|1|off\n", argv[0]); -	} - -        fp = open("/dev/usrp_e0", O_RDWR); -        printf("fp = %d\n", fp); -	if (fp < 0) { -		perror("Open failed"); -		return -1; -	} - -	if (strcmp(argv[1], "0") == 0) { -		printf("Selected 0 based on %s\n", argv[1]); -		write_reg(UE_REG_GPIO_TX_DDR, 0xFFFF); -		write_reg(UE_REG_GPIO_RX_DDR, 0xFFFF); -		write_reg(UE_REG_GPIO_TX_SEL, 0x0); -		write_reg(UE_REG_GPIO_RX_SEL, 0x0); -		write_reg(UE_REG_GPIO_TX_DBG, 0xFFFF); -		write_reg(UE_REG_GPIO_RX_DBG, 0xFFFF); -	} else if (strcmp(argv[1], "1") == 0) { -		printf("Selected 1 based on %s\n", argv[1]); -		write_reg(UE_REG_GPIO_TX_DDR, 0xFFFF); -		write_reg(UE_REG_GPIO_RX_DDR, 0xFFFF); -		write_reg(UE_REG_GPIO_TX_SEL, 0xFFFF); -		write_reg(UE_REG_GPIO_RX_SEL, 0xFFFF); -		write_reg(UE_REG_GPIO_TX_DBG, 0xFFFF); -		write_reg(UE_REG_GPIO_RX_DBG, 0xFFFF); -	} else { -		printf("Selected off based on %s\n", argv[1]); -		write_reg(UE_REG_GPIO_TX_DDR, 0x0); -		write_reg(UE_REG_GPIO_RX_DDR, 0x0); -	} - -	return 0; -} diff --git a/host/apps/omap_debug/usrp-e-i2c.c b/host/apps/omap_debug/usrp-e-i2c.c deleted file mode 100644 index da8709ae1..000000000 --- a/host/apps/omap_debug/usrp-e-i2c.c +++ /dev/null @@ -1,87 +0,0 @@ -#include <stdio.h> -#include <stdlib.h> -#include <string.h> -#include <unistd.h> -#include <sys/types.h> -#include <fcntl.h> -#include <sys/ioctl.h> - -#include "usrp_e.h" - -// Usage: usrp_e_i2c w address data0 data1 data 2 .... -// Usage: usrp_e_i2c r address count - -int main(int argc, char *argv[]) -{ -	int fp, ret, i, tmp; -	struct usrp_e_i2c *i2c_msg; -	int direction, address, count; - -	if (argc < 3) { -		printf("Usage: usrp-e-i2c w address data0 data1 data2 ...\n"); -		printf("Usage: usrp-e-i2c r address count\n"); -		printf("All addresses and data in hex.\n"); -		exit(-1); -	} - -	if (strcmp(argv[1], "r") == 0) { -		direction = 0; -	} else if (strcmp(argv[1], "w") == 0) { -		direction = 1; -	} else { -		return -1; -	} - -	sscanf(argv[2], "%X", &address); -	printf("Address = %X\n", address); - -	fp = open("/dev/usrp_e0", O_RDWR); -	printf("fp = %d\n", fp); -	if (fp < 0) { -		perror("Open failed"); -		return -1; -	} - -//	sleep(1); - -	if (direction) { -		count = argc - 3; -	} else { -		sscanf(argv[3], "%X", &count); -	} -	printf("Count = %X\n", count); - -	i2c_msg = malloc(sizeof(i2c_msg) + count * sizeof(char)); - -	i2c_msg->addr = address; -	i2c_msg->len = count; - -	for (i = 0; i < count; i++) { -		i2c_msg->data[i] = i; -	} - -	if (direction) { -		// Write - -		for (i=0; i<count; i++) { -			sscanf(argv[3+i], "%X", &tmp); -			i2c_msg->data[i] = tmp; -		} - -		ret = ioctl(fp, USRP_E_I2C_WRITE, i2c_msg); -		printf("Return value from i2c_write ioctl: %d\n", ret); -	} else { -		// Read - -		ret = ioctl(fp, USRP_E_I2C_READ, i2c_msg); -		printf("Return value from i2c_read ioctl: %d\n", ret); - -		printf("Ioctl: %d Data read :", ret); -		for (i=0; i<count; i++) { -			printf(" %X", i2c_msg->data[i]); -		} -		printf("\n"); -			 -	} -	return 0; -} diff --git a/host/apps/omap_debug/usrp-e-lb-test.c b/host/apps/omap_debug/usrp-e-lb-test.c deleted file mode 100644 index 68848064e..000000000 --- a/host/apps/omap_debug/usrp-e-lb-test.c +++ /dev/null @@ -1,58 +0,0 @@ -#include <stdio.h> -#include <sys/types.h> -#include <fcntl.h> -#include <pthread.h> -#include <stdlib.h> -#include <unistd.h> -#include <stddef.h> -#include "usrp_e.h" - -// max length #define PKT_DATA_LENGTH 1016 - -int main(int argc, char *argv[]) -{ -	struct usrp_transfer_frame *tx_data, *rx_data; -	int i, fp, packet_data_length, cnt; -	struct usrp_e_ctl16 d; - -	if (argc < 2) { -		printf("%s data_size (in bytes < 2040)\n", argv[0]); -		return -1; -	} - -	packet_data_length = atoi(argv[1]); - -	fp = open("/dev/usrp_e0", O_RDWR); - -	d.offset = 14; -	d.count = 1; -	d.buf[0] = (1 << 13); -	ioctl(fp, USRP_E_WRITE_CTL16, &d); - -	tx_data = malloc(2048); -	rx_data = malloc(2048); - -	tx_data->status = 0; -	tx_data->len = sizeof(struct usrp_transfer_frame) + packet_data_length; - -	while (1) { - -		for (i = 0; i < packet_data_length; i++) { -			tx_data->buf[i] = random() >> 24; - -		} - -		cnt = write(fp, tx_data, 2048); -		cnt = read(fp, rx_data, 2048); - -		if (tx_data->len != rx_data->len) -			printf("Bad frame length sent %d, read %d\n", tx_data->len, rx_data->len); - -		for (i = 0; i < packet_data_length; i++) { -			if (tx_data->buf[i] != rx_data->buf[i]) -				printf("Bad data at %d, sent %d, received %d\n", i, tx_data->buf[i], rx_data->buf[i]); -		} -		printf("---------------------------------------------------\n"); -		sleep(1); -	} -} diff --git a/host/apps/omap_debug/usrp-e-led.c b/host/apps/omap_debug/usrp-e-led.c deleted file mode 100644 index d1b6c8996..000000000 --- a/host/apps/omap_debug/usrp-e-led.c +++ /dev/null @@ -1,35 +0,0 @@ -#include <stdio.h> -#include <stdlib.h> -#include <sys/types.h> -#include <fcntl.h> -#include <string.h> -#include <sys/ioctl.h> -#include <unistd.h> - -#include "usrp_e.h" -#include "usrp_e_regs.hpp" - -// Usage: usrp_e_uart <string> - - -int main(int argc, char *argv[]) -{ -	int fp, i, ret; -	struct usrp_e_ctl16 d; - -	fp = open("/dev/usrp_e0", O_RDWR); -	printf("fp = %d\n", fp); - -	d.offset = UE_REG_MISC_BASE; -	d.count = 1; - -	while (1) { -		for (i=0; i<8; i++) { -			d.buf[0] = i; -			ret = ioctl(fp, USRP_E_WRITE_CTL16, &d); -			sleep(1); -		} -	} - -	return 0; -} diff --git a/host/apps/omap_debug/usrp-e-ram.c b/host/apps/omap_debug/usrp-e-ram.c deleted file mode 100644 index d548f7ccd..000000000 --- a/host/apps/omap_debug/usrp-e-ram.c +++ /dev/null @@ -1,25 +0,0 @@ -#include <stdio.h> -#include <sys/types.h> -#include <fcntl.h> - -int main(int rgc, char *argv[]) -{ -	int fp, i, cnt; -	unsigned short buf[1024]; -	unsigned short buf_rb[1024]; - -	fp = open("/dev/usrp1_e0", O_RDWR); -	printf("fp = %d\n", fp); - -	for (i=0; i<1024; i++) -		buf[i] = i*256;  -	write(fp, buf, 2048); -	read(fp, buf_rb, 2048); - -	printf("Read back %hX %hX\n", buf_rb[0], buf_rb[1]); - -	for (i=0; i<1024; i++) { -		if (buf[i] != buf_rb[i]) -			printf("Read - %hX, expected - %hX\n", buf_rb[i], buf[i]); -	} -} diff --git a/host/apps/omap_debug/usrp-e-read.c b/host/apps/omap_debug/usrp-e-read.c deleted file mode 100644 index c28f018d5..000000000 --- a/host/apps/omap_debug/usrp-e-read.c +++ /dev/null @@ -1,18 +0,0 @@ -#include <stdio.h> -#include <sys/types.h> -#include <fcntl.h> - -int main(int rgc, char *argv[]) -{ -	int fp, cnt; -	short buf[1024]; - -	fp = open("/dev/usrp1_e0", O_RDONLY); -	printf("fp = %d\n", fp); - -	do { -	cnt = read(fp, buf, 2048); -//	printf("Bytes read - %d\n", cnt); -	} while(1); -	printf("Data - %hX\n", buf[0]); -} diff --git a/host/apps/omap_debug/usrp-e-spi.c b/host/apps/omap_debug/usrp-e-spi.c deleted file mode 100644 index c353c409b..000000000 --- a/host/apps/omap_debug/usrp-e-spi.c +++ /dev/null @@ -1,54 +0,0 @@ -#include <stdio.h> -#include <stdlib.h> -#include <unistd.h> -#include <sys/types.h> -#include <fcntl.h> -#include <sys/ioctl.h> - -#include "usrp_e.h" - -// Usage: usrp_e_spi w|rb slave data - -int main(int argc, char *argv[]) -{ -	int fp, slave, length, ret; -	unsigned int data; -	struct usrp_e_spi spi_dat; - -	if (argc < 5) { -		printf("Usage: usrp_e_spi w|rb slave transfer_length data\n"); -		exit(-1); -	} - -	slave = atoi(argv[2]); -	length = atoi(argv[3]); -	data = atoll(argv[4]); - -	printf("Data = %X\n", data); - -	fp = open("/dev/usrp_e0", O_RDWR); -	printf("fp = %d\n", fp); -	if (fp < 0) { -		perror("Open failed"); -		return -1; -	} - -//	sleep(1); - - -	spi_dat.slave = slave; -	spi_dat.data = data; -	spi_dat.length = length; -	spi_dat.flags = UE_SPI_PUSH_FALL | UE_SPI_LATCH_RISE; - -	if (*argv[1] == 'r') { -		spi_dat.readback = 1; -		ret = ioctl(fp, USRP_E_SPI, &spi_dat); -		printf("Ioctl returns: %d, Data returned = %d\n", ret, spi_dat.data); -	} else { -		spi_dat.readback = 0; -		ioctl(fp, USRP_E_SPI, &spi_dat); -	} - -	return 0; -} diff --git a/host/apps/omap_debug/usrp-e-uart-rx.c b/host/apps/omap_debug/usrp-e-uart-rx.c deleted file mode 100644 index 24b417980..000000000 --- a/host/apps/omap_debug/usrp-e-uart-rx.c +++ /dev/null @@ -1,53 +0,0 @@ -#include <stdio.h> -#include <stdlib.h> -#include <sys/types.h> -#include <fcntl.h> -#include <string.h> -#include <sys/ioctl.h> - -#include "usrp_e.h" -#include "usrp_e_regs.hpp" - -// Usage: usrp_e_uart <string> - - -int main(int argc, char *argv[]) -{ -	int fp, ret; -	struct usrp_e_ctl16 d; -	__u16 clkdiv; - -	if (argc == 0) { -		printf("Usage: usrp-e-uart-rx <opt clkdiv>\n"); -		printf("clkdiv = 278 is 230.4k \n"); -		printf("clkdiv = 556 is 115.2k \n"); -		exit(-1); -	} - -	fp = open("/dev/usrp_e0", O_RDWR); -	printf("fp = %d\n", fp); - -	if (argc == 2) { -		clkdiv = atoi(argv[1]); -		d.offset = UE_REG_UART_CLKDIV; -		d.count = 1; -		d.buf[0] = clkdiv; -		ret = ioctl(fp, USRP_E_WRITE_CTL16, &d); -	} - -	while(1) { -		d.offset = UE_REG_UART_RXLEVEL; -		d.count = 1; -		ret = ioctl(fp, USRP_E_READ_CTL16, &d); - -		if (d.buf[0] > 0) { -			d.offset = UE_REG_UART_RXCHAR; -			d.count = 1; -			ret = ioctl(fp, USRP_E_READ_CTL16, &d); -			printf("%c", d.buf[0]); -			fflush(stdout); -		} -	} - -	return 0; -} diff --git a/host/apps/omap_debug/usrp-e-uart.c b/host/apps/omap_debug/usrp-e-uart.c deleted file mode 100644 index 2956c407f..000000000 --- a/host/apps/omap_debug/usrp-e-uart.c +++ /dev/null @@ -1,48 +0,0 @@ -#include <stdio.h> -#include <stdlib.h> -#include <sys/types.h> -#include <fcntl.h> -#include <string.h> -#include <sys/ioctl.h> - -#include "usrp_e.h" -#include "usrp_e_regs.hpp" - -// Usage: usrp_e_uart <string> - - -int main(int argc, char *argv[]) -{ -	int fp, i, ret; -	struct usrp_e_ctl16 d; -	char *str = argv[1]; -	__u16 clkdiv; - -	if (argc < 2) { -		printf("Usage: usrp_e_uart <string> <opt clkdiv>\n"); -		printf("clkdiv = 278 is 230.4k \n"); -		printf("clkdiv = 556 is 115.2k \n"); -		exit(-1); -	} - -	fp = open("/dev/usrp_e0", O_RDWR); -	printf("fp = %d\n", fp); - -	if (argc == 3) { -		clkdiv = atoi(argv[2]); -		d.offset = UE_REG_UART_CLKDIV; -		d.count = 1; -		d.buf[0] = clkdiv; -		ret = ioctl(fp, USRP_E_WRITE_CTL16, &d); -	} - -	for (i=0; i<strlen(str); i++) { -		d.offset = UE_REG_UART_TXCHAR; -		d.count = 1; -		d.buf[0] = str[i]; -		ret = ioctl(fp, USRP_E_WRITE_CTL16, &d); -		printf("Wrote %X, to %X, ret = %d\n", d.buf[0], d.offset, ret); -	} - -	return 0; -} diff --git a/host/apps/omap_debug/usrp-e-write.c b/host/apps/omap_debug/usrp-e-write.c deleted file mode 100644 index 903c0071f..000000000 --- a/host/apps/omap_debug/usrp-e-write.c +++ /dev/null @@ -1,21 +0,0 @@ -#include <stdio.h> -#include <sys/types.h> -#include <fcntl.h> - -int main(int rgc, char *argv[]) -{ -	int fp, i, cnt; -	short buf[1024]; - -	fp = open("/dev/usrp1_e0", O_WRONLY); -	printf("fp = %d\n", fp); - -	for (i=0; i<1024; i++) { -		buf[i] = i; -	} - -//	do { -		cnt = write(fp, buf, 2048); -		printf("Bytes written - %d\n", cnt); -//	} while (1); -} diff --git a/host/apps/omap_debug/usrp_e.h b/host/apps/omap_debug/usrp_e.h deleted file mode 100644 index 2c4aa2ac1..000000000 --- a/host/apps/omap_debug/usrp_e.h +++ /dev/null @@ -1,60 +0,0 @@ - -/* - *  Copyright (C) 2010 Ettus Research, LLC - * - *  Written by Philip Balister <philip@opensdr.com> - * - *  This program is free software; you can redistribute it and/or modify - *  it under the terms of the GNU General Public License as published by - *  the Free Software Foundation; either version 2 of the License, or - *  (at your option) any later version. - */ - -#ifndef __USRP_E_H -#define __USRP_E_H - -#include <linux/types.h> -#include <linux/ioctl.h> - -struct usrp_e_ctl16 { -	__u32 offset; -	__u32 count; -	__u16 buf[20]; -}; - -struct usrp_e_ctl32 { -	__u32 offset; -	__u32 count; -	__u32 buf[10]; -}; - -#define USRP_E_IOC_MAGIC	'u' -#define USRP_E_WRITE_CTL16	_IOW(USRP_E_IOC_MAGIC, 0x20, struct usrp_e_ctl16) -#define USRP_E_READ_CTL16	_IOWR(USRP_E_IOC_MAGIC, 0x21, struct usrp_e_ctl16) -#define USRP_E_WRITE_CTL32	_IOW(USRP_E_IOC_MAGIC, 0x22, struct usrp_e_ctl32) -#define USRP_E_READ_CTL32	_IOWR(USRP_E_IOC_MAGIC, 0x23, struct usrp_e_ctl32) -#define USRP_E_GET_RB_INFO      _IOR(USRP_E_IOC_MAGIC, 0x27, struct usrp_e_ring_buffer_size_t) -#define USRP_E_GET_COMPAT_NUMBER _IO(USRP_E_IOC_MAGIC, 0x28) - -#define USRP_E_COMPAT_NUMBER 2 - -/* Flag defines */ -#define RB_USER (1<<0) -#define RB_KERNEL (1<<1) -#define RB_OVERRUN (1<<2) -#define RB_DMA_ACTIVE (1<<3) -#define RB_USER_PROCESS (1<<4) - -struct ring_buffer_info { -	int flags; -	int len; -}; - -struct usrp_e_ring_buffer_size_t { -	int num_pages_rx_flags; -	int num_rx_frames; -	int num_pages_tx_flags; -	int num_tx_frames; -}; - -#endif diff --git a/host/docs/dboards.dox b/host/docs/dboards.dox index 13f3ccadd..b38fa5ae1 100644 --- a/host/docs/dboards.dox +++ b/host/docs/dboards.dox @@ -331,6 +331,10 @@ Sensors:  -   **rssi**: float for measured RSSI in dBm  -   **temperature**: float for measured temperature in degC +\subsection dboards_e300 E310 MIMO XCVR board + +Please refer to \ref e3x0_dboard_e310. +  \subsection dboards_dbsrxmod DBSRX - Modifying for other boards that USRP1  Due to different clocking capabilities, the DBSRX will require diff --git a/host/docs/usrp_e3x0.dox b/host/docs/usrp_e3x0.dox index 289ea3410..dd26e047e 100644 --- a/host/docs/usrp_e3x0.dox +++ b/host/docs/usrp_e3x0.dox @@ -132,7 +132,7 @@ which should return 'arm-oe-linux-gnueabi'.  -# Setup your environment as described in \ref e3x0_sdk_usage  -# Type the following in the build directory (assuming a build in host/build): -        $ cmake -DCMAKE_TOOLCHAIN_FILE=<youruhdsrc>/host/cmake/Toolchains/oe-sdk_cross.cmake -DENABLE_E300 .. +        $ cmake -DCMAKE_TOOLCHAIN_FILE=<youruhdsrc>/host/cmake/Toolchains/oe-sdk_cross.cmake -DENABLE_E300=On ..          $ make  \subsubsection e3x0_sdk_usage_gnuradio Building GNU Radio @@ -191,7 +191,7 @@ builds)   $ bitbake gnuradio-dev-image  \endcode -When this completes, the files needed to create the sd card are in +When this completes, the files needed to create the SD card are in  `tmp-glibc/deploy/images/ettus-e300`  -# Build the toolchain. @@ -395,7 +395,7 @@ usrp->set_rx_subdev_spec("A:A A:B");  The following sensors are available for the USRP-E Series motherboards;  they can be queried through the API. -- **fe_locked** - rx / tx frontend pll locked +- **fe_locked** - rx / tx frontend PLL locked  - **temp** - processor temperature value  - **gps_time** and **gps_locked** sensors are added when the GPSDO is found @@ -490,7 +490,7 @@ by pinging the USRP and making sure the LEDs start to blink.   - Communication    -# How do I enable X forwarding so I can run X apps on the e3x0?\n -     In the file `/etc/ssh/sshd_config`, unmcomment the line `#X11Forwarding no` +     In the file `/etc/ssh/sshd_config`, uncomment the line `#X11Forwarding no`       and change "no" to "yes".  \section e3x0_apps Applications diff --git a/host/examples/CMakeLists.txt b/host/examples/CMakeLists.txt index 1e6f2f013..598e42302 100644 --- a/host/examples/CMakeLists.txt +++ b/host/examples/CMakeLists.txt @@ -29,7 +29,6 @@ SET(example_sources      test_messages.cpp      test_pps_input.cpp      test_timed_commands.cpp -    transport_hammer.cpp      tx_bursts.cpp      tx_samples_from_file.cpp      tx_timed_samples.cpp diff --git a/host/examples/benchmark_rate.cpp b/host/examples/benchmark_rate.cpp index 03d8f3477..7ff8b9939 100644 --- a/host/examples/benchmark_rate.cpp +++ b/host/examples/benchmark_rate.cpp @@ -43,7 +43,12 @@ unsigned long long num_seq_errors = 0;  /***********************************************************************   * Benchmark RX Rate   **********************************************************************/ -void benchmark_rx_rate(uhd::usrp::multi_usrp::sptr usrp, const std::string &rx_cpu, uhd::rx_streamer::sptr rx_stream){ +void benchmark_rx_rate( +        uhd::usrp::multi_usrp::sptr usrp, +        const std::string &rx_cpu, +        uhd::rx_streamer::sptr rx_stream, +        bool random_nsamps +) {      uhd::set_thread_priority_safe();      //print pre-test summary @@ -68,15 +73,19 @@ void benchmark_rx_rate(uhd::usrp::multi_usrp::sptr usrp, const std::string &rx_c      rx_stream->issue_stream_cmd(cmd);      while (not boost::this_thread::interruption_requested()){ +        if (random_nsamps) { +            cmd.num_samps = rand() % max_samps_per_packet; +            rx_stream->issue_stream_cmd(cmd); +        }          try { -          num_rx_samps += rx_stream->recv(buffs, max_samps_per_packet, md)*rx_stream->get_num_channels(); +            num_rx_samps += rx_stream->recv(buffs, max_samps_per_packet, md)*rx_stream->get_num_channels();          }          catch (...) { -          /* apparently, the boost thread interruption can sometimes result in -             throwing exceptions not of type boost::exception, this catch allows -             this thread to still attempt to issue the STREAM_MODE_STOP_CONTINUOUS -          */ -          break; +            /* apparently, the boost thread interruption can sometimes result in +               throwing exceptions not of type boost::exception, this catch allows +               this thread to still attempt to issue the STREAM_MODE_STOP_CONTINUOUS +            */ +            break;          }          //handle the error codes @@ -109,7 +118,12 @@ void benchmark_rx_rate(uhd::usrp::multi_usrp::sptr usrp, const std::string &rx_c  /***********************************************************************   * Benchmark TX Rate   **********************************************************************/ -void benchmark_tx_rate(uhd::usrp::multi_usrp::sptr usrp, const std::string &tx_cpu, uhd::tx_streamer::sptr tx_stream){ +void benchmark_tx_rate( +        uhd::usrp::multi_usrp::sptr usrp, +        const std::string &tx_cpu, +        uhd::tx_streamer::sptr tx_stream, +        bool random_nsamps=false +) {      uhd::set_thread_priority_safe();      //print pre-test summary @@ -127,9 +141,25 @@ void benchmark_tx_rate(uhd::usrp::multi_usrp::sptr usrp, const std::string &tx_c          buffs.push_back(&buff.front()); //same buffer for each channel      md.has_time_spec = (buffs.size() != 1); -    while (not boost::this_thread::interruption_requested()){ -        num_tx_samps += tx_stream->send(buffs, max_samps_per_packet, md)*tx_stream->get_num_channels();; -        md.has_time_spec = false; +    if (random_nsamps) { +        std::srand( time(NULL) ); +        while(not boost::this_thread::interruption_requested()){ +            size_t total_num_samps = rand() % max_samps_per_packet; +            size_t num_acc_samps = 0; +            const float timeout = 1; + +            usrp->set_time_now(uhd::time_spec_t(0.0)); +            while(num_acc_samps < total_num_samps){ +                //send a single packet +                num_tx_samps += tx_stream->send(buffs, max_samps_per_packet, md, timeout)*tx_stream->get_num_channels(); +                num_acc_samps += std::min(total_num_samps-num_acc_samps, tx_stream->get_max_num_samps()); +            } +        } +    } else { +        while (not boost::this_thread::interruption_requested()){ +            num_tx_samps += tx_stream->send(buffs, max_samps_per_packet, md)*tx_stream->get_num_channels(); +            md.has_time_spec = false; +        }      }      //send a mini EOB packet @@ -182,6 +212,7 @@ int UHD_SAFE_MAIN(int argc, char *argv[]){      std::string rx_cpu, tx_cpu;      std::string mode;      std::string channel_list; +    bool random_nsamps = false;      //setup the program options      po::options_description desc("Allowed options"); @@ -196,6 +227,7 @@ int UHD_SAFE_MAIN(int argc, char *argv[]){          ("rx_cpu", po::value<std::string>(&rx_cpu)->default_value("fc32"), "specify the host/cpu sample mode for RX")          ("tx_cpu", po::value<std::string>(&tx_cpu)->default_value("fc32"), "specify the host/cpu sample mode for TX")          ("mode", po::value<std::string>(&mode)->default_value("none"), "multi-channel sync mode option: none, mimo") +        ("random", "Run with random values of samples in send() and recv() to stress-test the I/O.")          ("channels", po::value<std::string>(&channel_list)->default_value("0"), "which channel(s) to use (specify \"0\", \"1\", \"0,1\", etc)")      ;      po::variables_map vm; @@ -213,6 +245,12 @@ int UHD_SAFE_MAIN(int argc, char *argv[]){          return ~0;      } +    // Random number of samples? +    if (vm.count("random")) { +        std::cout << "Using random number of samples in send() and recv() calls." << std::endl; +        random_nsamps = true; +    } +      //create a usrp device      std::cout << std::endl;      uhd::device_addrs_t device_addrs = uhd::device::find(args, uhd::device::USRP); @@ -251,7 +289,7 @@ int UHD_SAFE_MAIN(int argc, char *argv[]){          uhd::stream_args_t stream_args(rx_cpu, rx_otw);          stream_args.channels = channel_nums;          uhd::rx_streamer::sptr rx_stream = usrp->get_rx_stream(stream_args); -        thread_group.create_thread(boost::bind(&benchmark_rx_rate, usrp, rx_cpu, rx_stream)); +        thread_group.create_thread(boost::bind(&benchmark_rx_rate, usrp, rx_cpu, rx_stream, random_nsamps));      }      //spawn the transmit test thread @@ -261,7 +299,7 @@ int UHD_SAFE_MAIN(int argc, char *argv[]){          uhd::stream_args_t stream_args(tx_cpu, tx_otw);          stream_args.channels = channel_nums;          uhd::tx_streamer::sptr tx_stream = usrp->get_tx_stream(stream_args); -        thread_group.create_thread(boost::bind(&benchmark_tx_rate, usrp, tx_cpu, tx_stream)); +        thread_group.create_thread(boost::bind(&benchmark_tx_rate, usrp, tx_cpu, tx_stream, random_nsamps));          thread_group.create_thread(boost::bind(&benchmark_tx_rate_async_helper, tx_stream));      } @@ -287,6 +325,5 @@ int UHD_SAFE_MAIN(int argc, char *argv[]){      //finished      std::cout << std::endl << "Done!" << std::endl << std::endl; -         return EXIT_SUCCESS;  } diff --git a/host/examples/rx_samples_to_file.cpp b/host/examples/rx_samples_to_file.cpp index fa3c21114..930e92125 100644 --- a/host/examples/rx_samples_to_file.cpp +++ b/host/examples/rx_samples_to_file.cpp @@ -56,7 +56,7 @@ template<typename samp_type> void recv_to_file(      std::vector<samp_type> buff(samps_per_buff);      std::ofstream outfile;      if (not null) -		outfile.open(file.c_str(), std::ofstream::binary); +        outfile.open(file.c_str(), std::ofstream::binary);      bool overflow_message = true;      //setup streaming @@ -78,8 +78,8 @@ template<typename samp_type> void recv_to_file(      typedef std::map<size_t,size_t> SizeMap;      SizeMap mapSizes; -    while(not stop_signal_called and (num_requested_samples != num_total_samps or num_requested_samples == 0)){ -		boost::system_time now = boost::get_system_time(); +    while(not stop_signal_called and (num_requested_samples != num_total_samps or num_requested_samples == 0)) { +        boost::system_time now = boost::get_system_time();          size_t num_rx_samps = rx_stream->recv(&buff.front(), buff.size(), md, 3.0, enable_size_map); @@ -88,7 +88,7 @@ template<typename samp_type> void recv_to_file(              break;          }          if (md.error_code == uhd::rx_metadata_t::ERROR_CODE_OVERFLOW){ -            if (overflow_message){ +            if (overflow_message) {                  overflow_message = false;                  std::cerr << boost::format(                      "Got an overflow indication. Please consider the following:\n" @@ -110,99 +110,99 @@ template<typename samp_type> void recv_to_file(                  throw std::runtime_error(error);          } -        if (enable_size_map){ -			SizeMap::iterator it = mapSizes.find(num_rx_samps); -			if (it == mapSizes.end()) -				mapSizes[num_rx_samps] = 0; -			mapSizes[num_rx_samps] += 1; -		} +        if (enable_size_map) { +            SizeMap::iterator it = mapSizes.find(num_rx_samps); +            if (it == mapSizes.end()) +                mapSizes[num_rx_samps] = 0; +            mapSizes[num_rx_samps] += 1; +        }          num_total_samps += num_rx_samps; -		if (outfile.is_open()) -			outfile.write((const char*)&buff.front(), num_rx_samps*sizeof(samp_type)); - -		if (bw_summary){ -			last_update_samps += num_rx_samps; -			boost::posix_time::time_duration update_diff = now - last_update; -			if (update_diff.ticks() > boost::posix_time::time_duration::ticks_per_second()) { -				double t = (double)update_diff.ticks() / (double)boost::posix_time::time_duration::ticks_per_second(); -				double r = (double)last_update_samps / t; -				std::cout << boost::format("\t%f Msps") % (r/1e6) << std::endl; -				last_update_samps = 0; -				last_update = now; -			} -		} +        if (outfile.is_open()) +            outfile.write((const char*)&buff.front(), num_rx_samps*sizeof(samp_type)); + +        if (bw_summary) { +            last_update_samps += num_rx_samps; +            boost::posix_time::time_duration update_diff = now - last_update; +            if (update_diff.ticks() > boost::posix_time::time_duration::ticks_per_second()) { +                double t = (double)update_diff.ticks() / (double)boost::posix_time::time_duration::ticks_per_second(); +                double r = (double)last_update_samps / t; +                std::cout << boost::format("\t%f Msps") % (r/1e6) << std::endl; +                last_update_samps = 0; +                last_update = now; +            } +        }          ticks_diff = now - start; -		if (ticks_requested > 0){ -			if ((unsigned long long)ticks_diff.ticks() > ticks_requested) -				break; -		} +        if (ticks_requested > 0){ +            if ((unsigned long long)ticks_diff.ticks() > ticks_requested) +                break; +        }      }      stream_cmd.stream_mode = uhd::stream_cmd_t::STREAM_MODE_STOP_CONTINUOUS;      rx_stream->issue_stream_cmd(stream_cmd);      if (outfile.is_open()) -		outfile.close(); - -    if (stats){ -		std::cout << std::endl; - -		double t = (double)ticks_diff.ticks() / (double)boost::posix_time::time_duration::ticks_per_second(); -		std::cout << boost::format("Received %d samples in %f seconds") % num_total_samps % t << std::endl; -		double r = (double)num_total_samps / t; -		std::cout << boost::format("%f Msps") % (r/1e6) << std::endl; - -		if (enable_size_map) { -			std::cout << std::endl; -			std::cout << "Packet size map (bytes: count)" << std::endl; -			for (SizeMap::iterator it = mapSizes.begin(); it != mapSizes.end(); it++) -				std::cout << it->first << ":\t" << it->second << std::endl; -		} -	} +        outfile.close(); + +    if (stats) { +        std::cout << std::endl; + +        double t = (double)ticks_diff.ticks() / (double)boost::posix_time::time_duration::ticks_per_second(); +        std::cout << boost::format("Received %d samples in %f seconds") % num_total_samps % t << std::endl; +        double r = (double)num_total_samps / t; +        std::cout << boost::format("%f Msps") % (r/1e6) << std::endl; + +        if (enable_size_map) { +            std::cout << std::endl; +            std::cout << "Packet size map (bytes: count)" << std::endl; +            for (SizeMap::iterator it = mapSizes.begin(); it != mapSizes.end(); it++) +                std::cout << it->first << ":\t" << it->second << std::endl; +        } +    }  }  typedef boost::function<uhd::sensor_value_t (const std::string&)> get_sensor_fn_t;  bool check_locked_sensor(std::vector<std::string> sensor_names, const char* sensor_name, get_sensor_fn_t get_sensor_fn, double setup_time){ -	if (std::find(sensor_names.begin(), sensor_names.end(), sensor_name) == sensor_names.end()) -		return false; - -	boost::system_time start = boost::get_system_time(); -	boost::system_time first_lock_time; - -	std::cout << boost::format("Waiting for \"%s\": ") % sensor_name; -	std::cout.flush(); - -	while (true){ -		if ((not first_lock_time.is_not_a_date_time()) and -			(boost::get_system_time() > (first_lock_time + boost::posix_time::seconds(setup_time)))) -		{ -			std::cout << " locked." << std::endl; -			break; -		} -		if (get_sensor_fn(sensor_name).to_bool()){ -			if (first_lock_time.is_not_a_date_time()) -				first_lock_time = boost::get_system_time(); -			std::cout << "+"; -			std::cout.flush(); -		} -		else{ -			first_lock_time = boost::system_time();	//reset to 'not a date time' - -			if (boost::get_system_time() > (start + boost::posix_time::seconds(setup_time))){ -				std::cout << std::endl; -				throw std::runtime_error(str(boost::format("timed out waiting for consecutive locks on sensor \"%s\"") % sensor_name)); -			} -			std::cout << "_"; -			std::cout.flush(); -		} -		boost::this_thread::sleep(boost::posix_time::milliseconds(100)); -	} -	std::cout << std::endl; -	return true; +    if (std::find(sensor_names.begin(), sensor_names.end(), sensor_name) == sensor_names.end()) +        return false; + +    boost::system_time start = boost::get_system_time(); +    boost::system_time first_lock_time; + +    std::cout << boost::format("Waiting for \"%s\": ") % sensor_name; +    std::cout.flush(); + +    while (true) { +        if ((not first_lock_time.is_not_a_date_time()) and +                (boost::get_system_time() > (first_lock_time + boost::posix_time::seconds(setup_time)))) +        { +            std::cout << " locked." << std::endl; +            break; +        } +        if (get_sensor_fn(sensor_name).to_bool()){ +            if (first_lock_time.is_not_a_date_time()) +                first_lock_time = boost::get_system_time(); +            std::cout << "+"; +            std::cout.flush(); +        } +        else { +            first_lock_time = boost::system_time();	//reset to 'not a date time' + +            if (boost::get_system_time() > (start + boost::posix_time::seconds(setup_time))){ +                std::cout << std::endl; +                throw std::runtime_error(str(boost::format("timed out waiting for consecutive locks on sensor \"%s\"") % sensor_name)); +            } +            std::cout << "_"; +            std::cout.flush(); +        } +        boost::this_thread::sleep(boost::posix_time::milliseconds(100)); +    } +    std::cout << std::endl; +    return true;  }  int UHD_SAFE_MAIN(int argc, char *argv[]){ @@ -246,8 +246,12 @@ int UHD_SAFE_MAIN(int argc, char *argv[]){      po::notify(vm);      //print the help message -    if (vm.count("help")){ +    if (vm.count("help")) {          std::cout << boost::format("UHD RX samples to file %s") % desc << std::endl; +        std::cout +            << std::endl +            << "This application streams data from a single channel of a USRP device to a file.\n" +            << std::endl;          return ~0;      } @@ -258,7 +262,7 @@ int UHD_SAFE_MAIN(int argc, char *argv[]){      bool continue_on_bad_packet = vm.count("continue") > 0;      if (enable_size_map) -		std::cout << "Packet size tracking enabled - will only recv one packet at a time!" << std::endl; +        std::cout << "Packet size tracking enabled - will only recv one packet at a time!" << std::endl;      //create a usrp device      std::cout << std::endl; @@ -283,23 +287,23 @@ int UHD_SAFE_MAIN(int argc, char *argv[]){      std::cout << boost::format("Actual RX Rate: %f Msps...") % (usrp->get_rx_rate()/1e6) << std::endl << std::endl;      //set the center frequency -    if (vm.count("freq")){	//with default of 0.0 this will always be true -		std::cout << boost::format("Setting RX Freq: %f MHz...") % (freq/1e6) << std::endl; +    if (vm.count("freq")) { //with default of 0.0 this will always be true +        std::cout << boost::format("Setting RX Freq: %f MHz...") % (freq/1e6) << std::endl;          uhd::tune_request_t tune_request(freq);          if(vm.count("int-n")) tune_request.args = uhd::device_addr_t("mode_n=integer"); -		usrp->set_rx_freq(tune_request); -		std::cout << boost::format("Actual RX Freq: %f MHz...") % (usrp->get_rx_freq()/1e6) << std::endl << std::endl; -	} +        usrp->set_rx_freq(tune_request); +        std::cout << boost::format("Actual RX Freq: %f MHz...") % (usrp->get_rx_freq()/1e6) << std::endl << std::endl; +    }      //set the rf gain -    if (vm.count("gain")){ +    if (vm.count("gain")) {          std::cout << boost::format("Setting RX Gain: %f dB...") % gain << std::endl;          usrp->set_rx_gain(gain);          std::cout << boost::format("Actual RX Gain: %f dB...") % usrp->get_rx_gain() << std::endl << std::endl;      }      //set the IF filter bandwidth -    if (vm.count("bw")){ +    if (vm.count("bw")) {          std::cout << boost::format("Setting RX Bandwidth: %f MHz...") % bw << std::endl;          usrp->set_rx_bandwidth(bw);          std::cout << boost::format("Actual RX Bandwidth: %f MHz...") % usrp->get_rx_bandwidth() << std::endl << std::endl; @@ -312,12 +316,12 @@ int UHD_SAFE_MAIN(int argc, char *argv[]){      //check Ref and LO Lock detect      if (not vm.count("skip-lo")){ -		check_locked_sensor(usrp->get_rx_sensor_names(0), "lo_locked", boost::bind(&uhd::usrp::multi_usrp::get_rx_sensor, usrp, _1, 0), setup_time); -		if (ref == "mimo") -			check_locked_sensor(usrp->get_mboard_sensor_names(0), "mimo_locked", boost::bind(&uhd::usrp::multi_usrp::get_mboard_sensor, usrp, _1, 0), setup_time); -		if (ref == "external") -			check_locked_sensor(usrp->get_mboard_sensor_names(0), "ref_locked", boost::bind(&uhd::usrp::multi_usrp::get_mboard_sensor, usrp, _1, 0), setup_time); -	} +        check_locked_sensor(usrp->get_rx_sensor_names(0), "lo_locked", boost::bind(&uhd::usrp::multi_usrp::get_rx_sensor, usrp, _1, 0), setup_time); +        if (ref == "mimo") +            check_locked_sensor(usrp->get_mboard_sensor_names(0), "mimo_locked", boost::bind(&uhd::usrp::multi_usrp::get_mboard_sensor, usrp, _1, 0), setup_time); +        if (ref == "external") +            check_locked_sensor(usrp->get_mboard_sensor_names(0), "ref_locked", boost::bind(&uhd::usrp::multi_usrp::get_mboard_sensor, usrp, _1, 0), setup_time); +    }      if (total_num_samps == 0){          std::signal(SIGINT, &sig_int_handler); @@ -325,7 +329,7 @@ int UHD_SAFE_MAIN(int argc, char *argv[]){      }  #define recv_to_file_args(format) \ -	(usrp, format, wirefmt, file, spb, total_num_samps, total_time, bw_summary, stats, null, enable_size_map, continue_on_bad_packet) +    (usrp, format, wirefmt, file, spb, total_num_samps, total_time, bw_summary, stats, null, enable_size_map, continue_on_bad_packet)      //recv to file      if (type == "double") recv_to_file<std::complex<double> >recv_to_file_args("fc64");      else if (type == "float") recv_to_file<std::complex<float> >recv_to_file_args("fc32"); diff --git a/host/examples/rx_timed_samples.cpp b/host/examples/rx_timed_samples.cpp index 30535907f..20abd92fe 100644 --- a/host/examples/rx_timed_samples.cpp +++ b/host/examples/rx_timed_samples.cpp @@ -48,7 +48,6 @@ int UHD_SAFE_MAIN(int argc, char *argv[]){          ("rate", po::value<double>(&rate)->default_value(100e6/16), "rate of incoming samples")          ("dilv", "specify to disable inner-loop verbose")          ("channels", po::value<std::string>(&channel_list)->default_value("0"), "which channel(s) to use (specify \"0\", \"1\", \"0,1\", etc)") -          ;      po::variables_map vm;      po::store(po::parse_command_line(argc, argv, desc), vm); diff --git a/host/examples/test_pps_input.cpp b/host/examples/test_pps_input.cpp index 889c98a45..3e6c4ba9d 100644 --- a/host/examples/test_pps_input.cpp +++ b/host/examples/test_pps_input.cpp @@ -47,6 +47,11 @@ int UHD_SAFE_MAIN(int argc, char *argv[]){      //print the help message      if (vm.count("help")){          std::cout << boost::format("UHD Test PPS Input %s") % desc << std::endl; +        std::cout +                << std::endl +                << "Tests if the PPS input signal is working. Will throw an error if not." +                << std::endl +                << std::endl;          return ~0;      } diff --git a/host/examples/transport_hammer.cpp b/host/examples/transport_hammer.cpp deleted file mode 100644 index 32e344e3e..000000000 --- a/host/examples/transport_hammer.cpp +++ /dev/null @@ -1,280 +0,0 @@ -// -// Copyright 2012 Ettus Research LLC -// -// This program is free software: you can redistribute it and/or modify -// it under the terms of the GNU General Public License as published by -// the Free Software Foundation, either version 3 of the License, or -// (at your option) any later version. -// -// This program is distributed in the hope that it will be useful, -// but WITHOUT ANY WARRANTY; without even the implied warranty of -// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the -// GNU General Public License for more details. -// -// You should have received a copy of the GNU General Public License -// along with this program.  If not, see <http://www.gnu.org/licenses/>. -// - -#include <uhd/utils/thread_priority.hpp> -#include <uhd/convert.hpp> -#include <uhd/utils/safe_main.hpp> -#include <uhd/usrp/multi_usrp.hpp> -#include <boost/program_options.hpp> -#include <boost/format.hpp> -#include <boost/thread/thread.hpp> -#include <boost/math/special_functions/round.hpp> -#include <iostream> -#include <complex> - -namespace po = boost::program_options; - -/*********************************************************************** - * Test result variables - **********************************************************************/ -unsigned long long num_overflows = 0; -unsigned long long num_underflows = 0; -unsigned long long num_rx_samps = 0; -unsigned long long num_tx_samps = 0; -unsigned long long num_dropped_samps = 0; -unsigned long long num_seq_errors = 0; - -/*********************************************************************** - * RX Hammer - **********************************************************************/ -void rx_hammer(uhd::usrp::multi_usrp::sptr usrp, const std::string &rx_cpu, uhd::rx_streamer::sptr rx_stream){ -    uhd::set_thread_priority_safe(); - -    //print pre-test summary -    std::cout << boost::format( -        "Testing receive rate %f Msps" -    ) % (usrp->get_rx_rate()/1e6) << std::endl; - -    //setup variables and allocate buffer -    uhd::rx_metadata_t md; -    const size_t max_samps_per_packet = rx_stream->get_max_num_samps(); -    std::vector<char> buff(max_samps_per_packet*uhd::convert::get_bytes_per_item(rx_cpu)); -    std::vector<void *> buffs; -    for (size_t ch = 0; ch < rx_stream->get_num_channels(); ch++) -        buffs.push_back(&buff.front()); //same buffer for each channel -    bool had_an_overflow = false; -    uhd::time_spec_t last_time; -    const double rate = usrp->get_rx_rate(); -    double timeout = 1; - -    uhd::stream_cmd_t cmd(uhd::stream_cmd_t::STREAM_MODE_NUM_SAMPS_AND_DONE); -    cmd.time_spec = usrp->get_time_now() + uhd::time_spec_t(0.05); -    cmd.stream_now = (buffs.size() == 1); -    srand( time(NULL) ); - -    while (not boost::this_thread::interruption_requested()){ -        cmd.num_samps = rand() % 100000; -        rx_stream->issue_stream_cmd(cmd); -        num_rx_samps += rx_stream->recv(buffs, max_samps_per_packet, md, timeout, true); - -        //handle the error codes -        switch(md.error_code){ -        case uhd::rx_metadata_t::ERROR_CODE_NONE: -            if (had_an_overflow){ -                had_an_overflow = false; -                num_dropped_samps += boost::math::iround((md.time_spec - last_time).get_real_secs()*rate); -            } -            break; - -        case uhd::rx_metadata_t::ERROR_CODE_OVERFLOW: -            had_an_overflow = true; -            last_time = md.time_spec; -            if (!md.out_of_sequence) -                num_overflows++; -            break; - -        default: -            std::cerr << "Receiver error: " << md.strerror() << std::endl; -            std::cerr << "Unexpected error on recv, continuing..." << std::endl; -            break; -        } -    } -} - -/*********************************************************************** - * TX Hammer - **********************************************************************/ -void tx_hammer(uhd::usrp::multi_usrp::sptr usrp, const std::string &tx_cpu, uhd::tx_streamer::sptr tx_stream){ -    uhd::set_thread_priority_safe(); - -    uhd::tx_metadata_t md; -    const size_t max_samps_per_packet = tx_stream->get_max_num_samps(); -    std::vector<char> buff(max_samps_per_packet*uhd::convert::get_bytes_per_item(tx_cpu)); -    std::vector<void *> buffs; -    for (size_t ch = 0; ch < tx_stream->get_num_channels(); ch++) -        buffs.push_back(&buff.front()); //same buffer for each channel - -    //print pre-test summary -    std::cout << boost::format( -        "Testing transmit rate %f Msps" -    ) % (usrp->get_tx_rate()/1e6) << std::endl; - -    //setup variables and allocate buffer -    std::srand( time(NULL) ); -    while(not boost::this_thread::interruption_requested()){ -        size_t total_num_samps = rand() % 100000; -        size_t num_acc_samps = 0; -        float timeout = 1; - -        usrp->set_time_now(uhd::time_spec_t(0.0)); -        while(num_acc_samps < total_num_samps){ - -            //send a single packet -            num_tx_samps += tx_stream->send(buffs, max_samps_per_packet, md, timeout); - -            num_acc_samps += std::min(total_num_samps-num_acc_samps, tx_stream->get_max_num_samps()); -        } -        //send a mini EOB packet -        md.end_of_burst = true; -        tx_stream->send("", 0, md); -    } -} - -void tx_hammer_async_helper(uhd::tx_streamer::sptr tx_stream){ -    //setup variables and allocate buffer -    uhd::async_metadata_t async_md; - -    while (not boost::this_thread::interruption_requested()){ - -        if (not tx_stream->recv_async_msg(async_md)) continue; - -        //handle the error codes -        switch(async_md.event_code){ -        case uhd::async_metadata_t::EVENT_CODE_BURST_ACK: -            return; - -        case uhd::async_metadata_t::EVENT_CODE_UNDERFLOW: -        case uhd::async_metadata_t::EVENT_CODE_UNDERFLOW_IN_PACKET: -            num_underflows++; -            break; - -        case uhd::async_metadata_t::EVENT_CODE_SEQ_ERROR: -        case uhd::async_metadata_t::EVENT_CODE_SEQ_ERROR_IN_BURST: -            num_seq_errors++; -            break; - -        default: -            std::cerr << "Event code: " << async_md.event_code << std::endl; -            std::cerr << "Unexpected event on async recv, continuing..." << std::endl; -            break; -        } -    } -} - -/*********************************************************************** - * Main code + dispatcher - **********************************************************************/ -int UHD_SAFE_MAIN(int argc, char *argv[]){ -    uhd::set_thread_priority_safe(); - -    //variables to be set by po -    std::string args; -    double duration; -    double rx_rate, tx_rate; -    std::string rx_otw, tx_otw; -    std::string rx_cpu, tx_cpu; -    std::string mode; - -    //setup the program options -    po::options_description desc("Allowed options"); -    desc.add_options() -        ("help", "help message") -        ("args", po::value<std::string>(&args)->default_value(""), "single uhd device address args") -        ("duration", po::value<double>(&duration)->default_value(10.0), "if random, specify duration for the test in seconds") -        ("rx_rate", po::value<double>(&rx_rate), "specify to perform a RX rate test (sps)") -        ("tx_rate", po::value<double>(&tx_rate), "specify to perform a TX rate test (sps)") -        ("rx_otw", po::value<std::string>(&rx_otw)->default_value("sc16"), "specify the over-the-wire sample mode for RX") -        ("tx_otw", po::value<std::string>(&tx_otw)->default_value("sc16"), "specify the over-the-wire sample mode for TX") -        ("rx_cpu", po::value<std::string>(&rx_cpu)->default_value("fc32"), "specify the host/cpu sample mode for RX") -        ("tx_cpu", po::value<std::string>(&tx_cpu)->default_value("fc32"), "specify the host/cpu sample mode for TX") -        ("mode", po::value<std::string>(&mode)->default_value("none"), "multi-channel sync mode option: none, mimo") -    ; -    po::variables_map vm; -    po::store(po::parse_command_line(argc, argv, desc), vm); -    po::notify(vm); - -    //print the help message -    if (vm.count("help") or (vm.count("rx_rate") + vm.count("tx_rate")) == 0){ -        //std::cout << boost::format("UHD Transport Hammer - %s") % desc << std::endl; -        std::cout << -        "UHD Transport Hammer: a transport layer stress test that continuously\n" -        "calls for random amounts of TX and RX samples\n\n"; -        std::cout << desc << std::endl << -        "    Specify --rx_rate for a receive-only test.\n" -        "    Specify --tx_rate for a transmit-only test.\n" -        "    Specify both options for a full-duplex test.\n" -        << std::endl; -        return ~0; -    } - -    //create a usrp device -    std::cout << std::endl; -    uhd::device_addrs_t device_addrs = uhd::device::find(args, uhd::device::USRP); -    if (not device_addrs.empty() and device_addrs.at(0).get("type", "") == "usrp1"){ -        std::cerr << "*** Warning! ***" << std::endl; -        std::cerr << "Results will be inaccurate on USRP1 due to insufficient features.\n" << std::endl; -    } -    std::cout << boost::format("Creating the usrp device with: %s...") % args << std::endl; -    uhd::usrp::multi_usrp::sptr usrp = uhd::usrp::multi_usrp::make(args); -    std::cout << boost::format("Using Device: %s") % usrp->get_pp_string() << std::endl; - -    if (mode == "mimo"){ -        usrp->set_clock_source("mimo", 0); -        usrp->set_time_source("mimo", 0); -        boost::this_thread::sleep(boost::posix_time::seconds(1)); -    } - -    boost::thread_group thread_group; - -    //spawn the receive test thread -    if (vm.count("rx_rate")){ -        usrp->set_rx_rate(rx_rate); -        //create a receive streamer -        uhd::stream_args_t stream_args(rx_cpu, rx_otw); -        for (size_t ch = 0; ch < usrp->get_num_mboards(); ch++) //linear channel mapping -            stream_args.channels.push_back(ch); -        uhd::rx_streamer::sptr rx_stream = usrp->get_rx_stream(stream_args); -        thread_group.create_thread(boost::bind(&rx_hammer, usrp, rx_cpu, rx_stream)); -    } - -    //spawn the transmit test thread -    if (vm.count("tx_rate")){ -        usrp->set_tx_rate(tx_rate); -        //create a transmit streamer -        uhd::stream_args_t stream_args(tx_cpu, tx_otw); -        for (size_t ch = 0; ch < usrp->get_num_mboards(); ch++) //linear channel mapping -            stream_args.channels.push_back(ch); -        uhd::tx_streamer::sptr tx_stream = usrp->get_tx_stream(stream_args); -        thread_group.create_thread(boost::bind(&tx_hammer, usrp, tx_cpu, tx_stream)); -        thread_group.create_thread(boost::bind(&tx_hammer_async_helper, tx_stream)); -    } - -    //sleep for the required duration -    const long secs = long(duration); -    const long usecs = long((duration - secs)*1e6); -    boost::this_thread::sleep(boost::posix_time::seconds(secs) + boost::posix_time::microseconds(usecs)); - -    //interrupt and join the threads -    thread_group.interrupt_all(); -    thread_group.join_all(); - -    //print summary -    std::cout << std::endl << boost::format( -        "Transport Hammer summary:\n" -        "  Num received samples:    %u\n" -        "  Num dropped samples:     %u\n" -        "  Num overflows detected:  %u\n" -        "  Num transmitted samples: %u\n" -        "  Num sequence errors:     %u\n" -        "  Num underflows detected: %u\n" -    ) % num_rx_samps % num_dropped_samps % num_overflows % num_tx_samps % num_seq_errors % num_underflows << std::endl; - -    //finished -    std::cout << std::endl << "Done!" << std::endl << std::endl; - -    return EXIT_SUCCESS; -} diff --git a/host/examples/tx_bursts.cpp b/host/examples/tx_bursts.cpp index fec89a0e4..bb71d4581 100644 --- a/host/examples/tx_bursts.cpp +++ b/host/examples/tx_bursts.cpp @@ -148,7 +148,6 @@ int UHD_SAFE_MAIN(int argc, char *argv[]){              size_t num_tx_samps = tx_stream->send(                  buffs, samps_to_send, md, timeout              ); -                              //do not use time spec for subsequent packets              md.has_time_spec = false;              md.start_of_burst = false; diff --git a/host/examples/tx_waveforms.cpp b/host/examples/tx_waveforms.cpp index a233cecb1..73bcb13ac 100644 --- a/host/examples/tx_waveforms.cpp +++ b/host/examples/tx_waveforms.cpp @@ -15,6 +15,7 @@  // along with this program.  If not, see <http://www.gnu.org/licenses/>.  // +#include "wavetable.hpp"  #include <uhd/utils/thread_priority.hpp>  #include <uhd/utils/safe_main.hpp>  #include <uhd/utils/static.hpp> @@ -28,9 +29,7 @@  #include <boost/lexical_cast.hpp>  #include <boost/algorithm/string.hpp>  #include <iostream> -#include <complex>  #include <csignal> -#include <cmath>  namespace po = boost::program_options; @@ -41,52 +40,6 @@ static bool stop_signal_called = false;  void sig_int_handler(int){stop_signal_called = true;}  /*********************************************************************** - * Waveform generators - **********************************************************************/ -static const size_t wave_table_len = 8192; - -class wave_table_class{ -public: -    wave_table_class(const std::string &wave_type, const float ampl): -        _wave_table(wave_table_len) -    { -        //compute real wave table with 1.0 amplitude -        std::vector<double> real_wave_table(wave_table_len); -        if (wave_type == "CONST"){ -            for (size_t i = 0; i < wave_table_len; i++) -                real_wave_table[i] = 1.0; -        } -        else if (wave_type == "SQUARE"){ -            for (size_t i = 0; i < wave_table_len; i++) -                real_wave_table[i] = (i < wave_table_len/2)? 0.0 : 1.0; -        } -        else if (wave_type == "RAMP"){ -            for (size_t i = 0; i < wave_table_len; i++) -                real_wave_table[i] = 2.0*i/(wave_table_len-1) - 1.0; -        } -        else if (wave_type == "SINE"){ -            static const double tau = 2*std::acos(-1.0); -            for (size_t i = 0; i < wave_table_len; i++) -                real_wave_table[i] = std::sin((tau*i)/wave_table_len); -        } -        else throw std::runtime_error("unknown waveform type: " + wave_type); - -        //compute i and q pairs with 90% offset and scale to amplitude -        for (size_t i = 0; i < wave_table_len; i++){ -            const size_t q = (i+(3*wave_table_len)/4)%wave_table_len; -            _wave_table[i] = std::complex<float>(ampl*real_wave_table[i], ampl*real_wave_table[q]); -        } -    } - -    inline std::complex<float> operator()(const size_t index) const{ -        return _wave_table[index % wave_table_len]; -    } - -private: -    std::vector<std::complex<float> > _wave_table; -}; - -/***********************************************************************   * Main function   **********************************************************************/  int UHD_SAFE_MAIN(int argc, char *argv[]){ diff --git a/host/examples/txrx_loopback_to_file.cpp b/host/examples/txrx_loopback_to_file.cpp index 10d6b7af4..1c7973df9 100644 --- a/host/examples/txrx_loopback_to_file.cpp +++ b/host/examples/txrx_loopback_to_file.cpp @@ -15,6 +15,7 @@  // along with this program.  If not, see <http://www.gnu.org/licenses/>.  // +#include "wavetable.hpp"  #include <uhd/types/tune_request.hpp>  #include <uhd/utils/thread_priority.hpp>  #include <uhd/utils/safe_main.hpp> @@ -31,7 +32,6 @@  #include <iostream>  #include <fstream>  #include <csignal> -#include <cmath>  namespace po = boost::program_options; @@ -61,51 +61,6 @@ std::string generate_out_filename(const std::string &base_fn, size_t n_names, si      return base_fn_fp.string();  } -/*********************************************************************** - * Waveform generators - **********************************************************************/ -static const size_t wave_table_len = 8192; - -class wave_table_class{ -public: -    wave_table_class(const std::string &wave_type, const float ampl): -        _wave_table(wave_table_len) -    { -        //compute real wave table with 1.0 amplitude -        std::vector<double> real_wave_table(wave_table_len); -        if (wave_type == "CONST"){ -            for (size_t i = 0; i < wave_table_len; i++) -                real_wave_table[i] = 1.0; -        } -        else if (wave_type == "SQUARE"){ -            for (size_t i = 0; i < wave_table_len; i++) -                real_wave_table[i] = (i < wave_table_len/2)? 0.0 : 1.0; -        } -        else if (wave_type == "RAMP"){ -            for (size_t i = 0; i < wave_table_len; i++) -                real_wave_table[i] = 2.0*i/(wave_table_len-1) - 1.0; -        } -        else if (wave_type == "SINE"){ -            static const double tau = 2*std::acos(-1.0); -            for (size_t i = 0; i < wave_table_len; i++) -                real_wave_table[i] = std::sin((tau*i)/wave_table_len); -        } -        else throw std::runtime_error("unknown waveform type: " + wave_type); - -        //compute i and q pairs with 90% offset and scale to amplitude -        for (size_t i = 0; i < wave_table_len; i++){ -            const size_t q = (i+(3*wave_table_len)/4)%wave_table_len; -            _wave_table[i] = std::complex<float>(ampl*real_wave_table[i], ampl*real_wave_table[q]); -        } -    } - -    inline std::complex<float> operator()(const size_t index) const{ -        return _wave_table[index % wave_table_len]; -    } - -private: -    std::vector<std::complex<float> > _wave_table; -};  /***********************************************************************   * transmit_worker function diff --git a/host/examples/wavetable.hpp b/host/examples/wavetable.hpp new file mode 100644 index 000000000..d7ffc8406 --- /dev/null +++ b/host/examples/wavetable.hpp @@ -0,0 +1,66 @@ +// +// Copyright 2010-2012,2014 Ettus Research LLC +// +// This program is free software: you can redistribute it and/or modify +// it under the terms of the GNU General Public License as published by +// the Free Software Foundation, either version 3 of the License, or +// (at your option) any later version. +// +// This program is distributed in the hope that it will be useful, +// but WITHOUT ANY WARRANTY; without even the implied warranty of +// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the +// GNU General Public License for more details. +// +// You should have received a copy of the GNU General Public License +// along with this program.  If not, see <http://www.gnu.org/licenses/>. +// + +#include <string> +#include <cmath> +#include <complex> +#include <vector> +#include <stdexcept> + +static const size_t wave_table_len = 8192; + +class wave_table_class{ +public: +    wave_table_class(const std::string &wave_type, const float ampl): +        _wave_table(wave_table_len) +    { +        //compute real wave table with 1.0 amplitude +        std::vector<double> real_wave_table(wave_table_len); +        if (wave_type == "CONST"){ +            for (size_t i = 0; i < wave_table_len; i++) +                real_wave_table[i] = 1.0; +        } +        else if (wave_type == "SQUARE"){ +            for (size_t i = 0; i < wave_table_len; i++) +                real_wave_table[i] = (i < wave_table_len/2)? 0.0 : 1.0; +        } +        else if (wave_type == "RAMP"){ +            for (size_t i = 0; i < wave_table_len; i++) +                real_wave_table[i] = 2.0*i/(wave_table_len-1) - 1.0; +        } +        else if (wave_type == "SINE"){ +            static const double tau = 2*std::acos(-1.0); +            for (size_t i = 0; i < wave_table_len; i++) +                real_wave_table[i] = std::sin((tau*i)/wave_table_len); +        } +        else throw std::runtime_error("unknown waveform type: " + wave_type); + +        //compute i and q pairs with 90% offset and scale to amplitude +        for (size_t i = 0; i < wave_table_len; i++){ +            const size_t q = (i+(3*wave_table_len)/4)%wave_table_len; +            _wave_table[i] = std::complex<float>(ampl*real_wave_table[i], ampl*real_wave_table[q]); +        } +    } + +    inline std::complex<float> operator()(const size_t index) const{ +        return _wave_table[index % wave_table_len]; +    } + +private: +    std::vector<std::complex<float> > _wave_table; +}; + diff --git a/host/include/uhd/types/CMakeLists.txt b/host/include/uhd/types/CMakeLists.txt index 334f1b41b..66b8662a1 100644 --- a/host/include/uhd/types/CMakeLists.txt +++ b/host/include/uhd/types/CMakeLists.txt @@ -29,6 +29,7 @@ UHD_INSTALL(FILES      ref_vector.hpp      sensors.hpp      serial.hpp +    sid.hpp      stream_cmd.hpp      time_spec.hpp      tune_request.hpp diff --git a/host/include/uhd/types/sid.hpp b/host/include/uhd/types/sid.hpp new file mode 100644 index 000000000..12f98ff97 --- /dev/null +++ b/host/include/uhd/types/sid.hpp @@ -0,0 +1,238 @@ +// +// Copyright 2014 Ettus Research LLC +// +// This program is free software: you can redistribute it and/or modify +// it under the terms of the GNU General Public License as published by +// the Free Software Foundation, either version 3 of the License, or +// (at your option) any later version. +// +// This program is distributed in the hope that it will be useful, +// but WITHOUT ANY WARRANTY; without even the implied warranty of +// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the +// GNU General Public License for more details. +// +// You should have received a copy of the GNU General Public License +// along with this program.  If not, see <http://www.gnu.org/licenses/>. +// + +#ifndef INCLUDED_UHD_TYPES_SID_HPP +#define INCLUDED_UHD_TYPES_SID_HPP + +#include <uhd/config.hpp> +#include <boost/cstdint.hpp> +#include <boost/shared_ptr.hpp> +#include <iostream> + +namespace uhd { +    /*! +     * \brief Represents a stream ID (SID). +     * +     * A stream ID (SID) is an identifier for data. +     * It is a 32-Bit value which consistst of 16 Bits +     * for the source address and 16 Bits for the destination +     * address. +     * Every address is split into two parts: The _address_, which +     * identifies the device used, and the _endpoint_, which identifies +     * a specific object inside the given device (e.g., a block). +     * *Note:* In the case where there are several crossbars on a single +     * device, each crossbar gets its own address. +     * Both address and endpoint are 8 bits in length. If a 16-bit address +     * is required, we use the combination of the 8-bit address and the 8-bit +     * endpoint. +     * +     * \section sid_str_repr +     * +     * The string representation of a SID is of the form +     * +     *     2.3>0.6 +     * +     * The '>' symbol shows the direction, so in this case, +     * data is flowing from address 2.3 to 0.6. +     * +     * As a convention, ':' is used instead of '.' when giving the +     * SID in hexadecimal numbers, and two characters are used for each +     * address part. As an example, the following two SIDs are identical: +     * +     *     2.3>0.16 (decimal) +     *     02:03>00:10 (hexadecimal) +     * +     * The format is: +     *     SRC_ADDRESS.SRC_ENDPOINT>DST_ADDRESS.DST_ENDPOINT +     * +     * +     * \section sid_block_ports +     * +     * In the special case where a block on a crossbar is addressed, the +     * endpoint is further split up into two parts of four bits each: The +     * first four bits specify the port number on the crossbar, whereas the +     * lower four bits represent the *block port*. As an example, consider +     * the following SID, given in hexadecimal: +     * +     *    00:10>02:A1 +     * +     * In this example, assume data is flowing from the host computer to an +     * X300. The crossbar address is 02. The endpoint is A1, which means we +     * are accessing a block on crossbar port A (the tenth port), and are addressing +     * block port 1. +     * +     */ +    class UHD_API sid_t +    { +    public: +        //! Create an unset SID +        sid_t(); +        //! Create a sid_t object from a 32-Bit SID value +        sid_t(boost::uint32_t sid); +        //! Create a sid_t object from its four components +        sid_t(boost::uint8_t src_addr, boost::uint8_t src_ep, boost::uint8_t dst_addr, boost::uint8_t dst_ep); +        //! Convert a string representation of a SID into its numerical representation +        sid_t(const std::string &); + +        //! Return a decimal string representation of the SID. +        std::string to_pp_string() const; +        //! Return a hexadecimal string representation of the SID. +        std::string to_pp_string_hex() const; + +        //! Returns true if this actually holds a valid SID +        bool is_set() const { return _set; }; + +        // Getters +        // +        //! Alias for get_sid() +        UHD_INLINE boost::uint32_t get() const { return get_sid(); }; +        //! Returns a 32-Bit representation of the SID if set, or zero otherwise. +        UHD_INLINE boost::uint32_t get_sid() const { return _set ? _sid : 0; }; +        //! Return the 16-bit source address of this SID +        UHD_INLINE boost::uint32_t get_src() const { +            return (_sid >> 16) & 0xFFFF; +        } +        //! Return the 16-bit destination address of this SID +        UHD_INLINE boost::uint32_t get_dst() const { +            return _sid & 0xFFFF; +        } +        //! Return 8-bit address of the source +        UHD_INLINE boost::uint32_t get_src_addr() const { +            return (get_src() >> 8) & 0xFF; +        } +        //! Return endpoint of the source +        UHD_INLINE boost::uint32_t get_src_endpoint() const { +            return get_src() & 0xFF; +        } +        //! Return crossbar port of the source +        UHD_INLINE boost::uint32_t get_src_xbarport() const { +            return (get_src_endpoint() >> 4) & 0xF; +        } +        //! Return block port of the source +        UHD_INLINE boost::uint32_t get_src_blockport() const { +            return (get_src_endpoint()) & 0xF; +        } +        //! Return 8-bit address of the destination +        UHD_INLINE boost::uint32_t get_dst_addr() const { +            return (get_dst() >> 8) & 0xFF; +        } +        //! Return endpoint of the destination +        UHD_INLINE boost::uint32_t get_dst_endpoint() const { +            return get_dst() & 0xFF; +        } +        //! Return crossbar port of the source +        UHD_INLINE boost::uint32_t get_dst_xbarport() const { +            return (get_dst_endpoint() >> 4) & 0xF; +        } +        //! Return block port of the source +        UHD_INLINE boost::uint32_t get_dst_blockport() const { +            return (get_dst_endpoint()) & 0xF; +        } + +        // Setters + +        //! Alias for set_sid() +        void set(boost::uint32_t new_sid) { set_sid(new_sid); }; +        //! Convert a string representation of a SID into a numerical one +        // Throws uhd::value_error if the string is not a valid SID +        // representation. +        void set_from_str(const std::string &); +        void set_sid(boost::uint32_t new_sid); +        //! Set the source address of this SID +        //  (the first 16 Bits) +        void set_src(boost::uint32_t new_addr); +        //! Set the destination address of this SID +        //  (the last 16 Bits) +        void set_dst(boost::uint32_t new_addr); +        void set_src_addr(boost::uint32_t new_addr); +        void set_src_endpoint(boost::uint32_t new_addr); +        void set_dst_addr(boost::uint32_t new_addr); +        void set_dst_endpoint(boost::uint32_t new_addr); +        void set_dst_xbarport(boost::uint32_t new_xbarport); +        void set_dst_blockport(boost::uint32_t new_blockport); + +        // Manipulators + +        //! Swaps dst and src address and returns the new SID. +        sid_t reversed(); + +        //! Swaps dst and src in-place. +        void reverse(); + +        // Overloaded operators + +        sid_t operator = (boost::uint32_t new_sid) { +            set_sid(new_sid); +            return *this; +        } + +        sid_t operator = (sid_t &sid) { +            set_sid(sid.get_sid()); +            return *this; +        } + +        sid_t operator = (const std::string &sid_str) { +            set_from_str(sid_str); +            return *this; +        } + +        bool operator == (const sid_t &sid) const { +            return (not _set and not sid.is_set()) or (_sid == sid.get_sid()); +        } + +        bool operator == (boost::uint32_t sid) const { +            return _set and _sid == sid; +        } + +        bool operator == (const std::string &sid_str) const { +            sid_t rhs(sid_str); +            return *this == rhs; +        } + +        // overloaded type casts are tricky, but for now we'll need them +        // for backward compatibility. consider them deprecated. + +        //! If the SID is not set, always returns zero. +        //  Use is_set() to check if the return value is valid. +        operator boost::uint32_t() const { +            return get(); +        } + +        operator bool() const { +            return _set; +        } + +    private: +        boost::uint32_t _sid; +        bool _set; +    }; + +    //! Stream output operator. Honors std::ios::hex. +    UHD_INLINE std::ostream& operator<< (std::ostream& out, const sid_t &sid) { +        std::ios_base::fmtflags ff = out.flags(); +        if (ff & std::ios::hex) { +            out << sid.to_pp_string_hex(); +        } else { +            out << sid.to_pp_string(); +        } +        return out; +    } + +} //namespace uhd + +#endif /* INCLUDED_UHD_TYPES_SID_HPP */ +// vim: sw=4 et: diff --git a/host/lib/types/CMakeLists.txt b/host/lib/types/CMakeLists.txt index f19043c1e..853da3fe2 100644 --- a/host/lib/types/CMakeLists.txt +++ b/host/lib/types/CMakeLists.txt @@ -86,6 +86,7 @@ LIBUHD_APPEND_SOURCES(      ${CMAKE_CURRENT_SOURCE_DIR}/ranges.cpp      ${CMAKE_CURRENT_SOURCE_DIR}/sensors.cpp      ${CMAKE_CURRENT_SOURCE_DIR}/serial.cpp +    ${CMAKE_CURRENT_SOURCE_DIR}/sid.cpp      ${CMAKE_CURRENT_SOURCE_DIR}/time_spec.cpp      ${CMAKE_CURRENT_SOURCE_DIR}/tune.cpp      ${CMAKE_CURRENT_SOURCE_DIR}/types.cpp diff --git a/host/lib/types/sid.cpp b/host/lib/types/sid.cpp new file mode 100644 index 000000000..2fc3781cf --- /dev/null +++ b/host/lib/types/sid.cpp @@ -0,0 +1,153 @@ +// +// Copyright 2014 Ettus Research LLC +// +// This program is free software: you can redistribute it and/or modify +// it under the terms of the GNU General Public License as published by +// the Free Software Foundation, either version 3 of the License, or +// (at your option) any later version. +// +// This program is distributed in the hope that it will be useful, +// but WITHOUT ANY WARRANTY; without even the implied warranty of +// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the +// GNU General Public License for more details. +// +// You should have received a copy of the GNU General Public License +// along with this program.  If not, see <http://www.gnu.org/licenses/>. +// + +#include <boost/format.hpp> +#include <boost/regex.hpp> +#include <boost/lexical_cast.hpp> +#include <uhd/exception.hpp> +#include <uhd/types/sid.hpp> +#include <uhd/utils/cast.hpp> + +using namespace uhd; + +sid_t::sid_t() +    : _sid(0x0000), _set(false) +{ +} + +sid_t::sid_t(boost::uint32_t sid) +    : _sid(sid), _set(true) +{ +} + +sid_t::sid_t(boost::uint8_t src_addr, boost::uint8_t src_ep, boost::uint8_t dst_addr, boost::uint8_t dst_ep) +    :  _sid(0x0000), _set(true) +{ +    set_src_addr(src_addr); +    set_src_endpoint(src_ep); +    set_dst_addr(dst_addr); +    set_dst_endpoint(dst_ep); +} + +sid_t::sid_t(const std::string &sid_str) +    : _sid(0x0000), _set(false) +{ +    set_from_str(sid_str); +} + +std::string sid_t::to_pp_string() const +{ +    if (not _set) { +        return "x.x>x.x"; +    } +    return str(boost::format("%d.%d>%d.%d") +        % get_src_addr() +        % get_src_endpoint() +        % get_dst_addr() +        % get_dst_endpoint() +    ); +} + +std::string sid_t::to_pp_string_hex() const +{ +    if (not _set) { +        return "xx:xx>xx:xx"; +    } +    return str(boost::format("%02x:%02x>%02x:%02x") +        % get_src_addr() +        % get_src_endpoint() +        % get_dst_addr() +        % get_dst_endpoint() +    ); +} + + +void sid_t::set_sid(boost::uint32_t new_sid) +{ +    _set = true; +    _sid = new_sid; +} + +void sid_t::set_from_str(const std::string &sid_str) +{ +    const std::string dec_regex = "(\\d{1,3})\\.(\\d{1,3})[.:/><](\\d{1,3})\\.(\\d{1,3})"; +    const std::string hex_regex = "([[:xdigit:]]{2}):([[:xdigit:]]{2})[.:/><]([[:xdigit:]]{2}):([[:xdigit:]]{2})"; + +    boost::cmatch matches; +    if (boost::regex_match(sid_str.c_str(), matches, boost::regex(dec_regex))) { +        set_src_addr(boost::lexical_cast<size_t>(matches[1])); +        set_src_endpoint(boost::lexical_cast<size_t>(matches[2])); +        set_dst_addr(boost::lexical_cast<size_t>(matches[3])); +        set_dst_endpoint(boost::lexical_cast<size_t>(matches[4])); +        return; +    } + +    if (boost::regex_match(sid_str.c_str(), matches, boost::regex(hex_regex))) { +        set_src_addr(uhd::cast::hexstr_cast<size_t>(matches[1])); +        set_src_endpoint(uhd::cast::hexstr_cast<size_t>(matches[2])); +        set_dst_addr(uhd::cast::hexstr_cast<size_t>(matches[3])); +        set_dst_endpoint(uhd::cast::hexstr_cast<size_t>(matches[4])); +        return; +    } + +    throw uhd::value_error(str(boost::format("Invalid SID representation: %s") % sid_str)); +} + +void sid_t::set_src(boost::uint32_t new_addr) { +    set_sid((_sid & 0x0000FFFF) | ((new_addr & 0xFFFF) << 16)); +} + +void sid_t::set_dst(boost::uint32_t new_addr) { +    set_sid((_sid & 0xFFFF0000) | (new_addr & 0xFFFF)); +} + +void sid_t::set_src_addr(boost::uint32_t new_addr) { +    set_sid((_sid & 0x00FFFFFF) | ((new_addr & 0xFF) << 24)); +} + +void sid_t::set_src_endpoint(boost::uint32_t new_addr) { +    set_sid((_sid & 0xFF00FFFF) | ((new_addr & 0xFF) << 16)); +} + +void sid_t::set_dst_addr(boost::uint32_t new_addr) { +    set_sid((_sid & 0xFFFF00FF) | ((new_addr & 0xFF) << 8)); +} + +void sid_t::set_dst_endpoint(boost::uint32_t new_addr) { +    set_sid((_sid & 0xFFFFFF00) | ((new_addr & 0xFF) << 0)); +} + +void sid_t::set_dst_xbarport(boost::uint32_t new_xbarport) +{ +    set_sid((_sid & 0xFFFFFF0F) | ((new_xbarport & 0xF) << 4)); +} + +void sid_t::set_dst_blockport(boost::uint32_t new_blockport) +{ +    set_sid((_sid & 0xFFFFFFF0) | ((new_blockport & 0xF) << 0)); +} + +sid_t sid_t::reversed() +{ +    return sid_t((get_dst() << 16) | get_src()); +} + +void sid_t::reverse() +{ +    set_sid((get_dst() << 16) | get_src()); +} + diff --git a/host/tests/CMakeLists.txt b/host/tests/CMakeLists.txt index 62544b69b..ea0b30cb8 100644 --- a/host/tests/CMakeLists.txt +++ b/host/tests/CMakeLists.txt @@ -37,6 +37,7 @@ SET(test_sources      msg_test.cpp      property_test.cpp      ranges_test.cpp +    sid_t_test.cpp      sph_recv_test.cpp      sph_send_test.cpp      subdev_spec_test.cpp diff --git a/host/tests/sid_t_test.cpp b/host/tests/sid_t_test.cpp new file mode 100644 index 000000000..71ab2c213 --- /dev/null +++ b/host/tests/sid_t_test.cpp @@ -0,0 +1,158 @@ +// +// Copyright 2014 Ettus Research LLC +// +// This program is free software: you can redistribute it and/or modify +// it under the terms of the GNU General Public License as published by +// the Free Software Foundation, either version 3 of the License, or +// (at your option) any later version. +// +// This program is distributed in the hope that it will be useful, +// but WITHOUT ANY WARRANTY; without even the implied warranty of +// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the +// GNU General Public License for more details. +// +// You should have received a copy of the GNU General Public License +// along with this program.  If not, see <http://www.gnu.org/licenses/>. +// + +#include <iostream> +#include <sstream> +#include <boost/test/unit_test.hpp> +#include <uhd/types/sid.hpp> +#include <uhd/exception.hpp> + +using uhd::sid_t; + +BOOST_AUTO_TEST_CASE(test_sid_t) { +    boost::uint32_t sid_value = 0x01020310; +    sid_t sid(sid_value); + +    BOOST_CHECK_EQUAL(sid.is_set(), true); +    BOOST_CHECK_EQUAL(sid.to_pp_string(), "1.2>3.16"); +    BOOST_CHECK_EQUAL(sid.to_pp_string_hex(), "01:02>03:10"); +    BOOST_CHECK_EQUAL(sid.get_src(), 0x0102); +    BOOST_CHECK_EQUAL(sid.get_dst(), 0x0310); +    BOOST_CHECK_EQUAL(sid.get_src_addr(), 0x01); +    BOOST_CHECK_EQUAL(sid.get_src_endpoint(), 0x02); +    BOOST_CHECK_EQUAL(sid.get_dst_addr(), 0x03); +    BOOST_CHECK_EQUAL(sid.get_dst_endpoint(), 0x10); +    BOOST_CHECK_EQUAL(sid == sid, true); +    BOOST_CHECK_EQUAL(sid == sid_value, true); + +    boost::uint32_t check_sid_val = (boost::uint32_t) sid; +    BOOST_CHECK_EQUAL(check_sid_val, sid_value); + +    std::stringstream ss_dec; +    ss_dec << sid; +    BOOST_CHECK_EQUAL(ss_dec.str(), "1.2>3.16"); + +    std::stringstream ss_hex; +    ss_hex << std::hex << sid; +    BOOST_CHECK_EQUAL(ss_hex.str(), "01:02>03:10"); + +    sid_t empty_sid; +    BOOST_CHECK_EQUAL(empty_sid.is_set(), false); +    BOOST_CHECK_EQUAL(empty_sid.to_pp_string(), "x.x>x.x"); +    BOOST_CHECK_EQUAL(empty_sid.to_pp_string_hex(), "xx:xx>xx:xx"); +    BOOST_CHECK_EQUAL(empty_sid == sid, false); +    BOOST_CHECK_EQUAL(empty_sid == sid_value, false); +    BOOST_CHECK_EQUAL((bool) empty_sid, false); + +    empty_sid = sid_value; // No longer empty +    BOOST_CHECK_EQUAL(empty_sid.is_set(), true); +    BOOST_CHECK_EQUAL(empty_sid == sid, true); +} + +BOOST_AUTO_TEST_CASE(test_sid_t_set) { +    boost::uint32_t sid_value = 0x0; +    sid_t sid(sid_value); + +    sid.set(0x01020304); +    BOOST_CHECK_EQUAL(sid.get(), 0x01020304); +    BOOST_CHECK_EQUAL(sid.get_src_addr(), 0x01); +    BOOST_CHECK_EQUAL(sid.get_src_endpoint(), 0x02); +    BOOST_CHECK_EQUAL(sid.get_dst_addr(), 0x03); +    BOOST_CHECK_EQUAL(sid.get_dst_endpoint(), 0x04); +    BOOST_CHECK_EQUAL(sid.get_dst_xbarport(), 0x0); +    BOOST_CHECK_EQUAL(sid.get_dst_blockport(), 0x4); + +    sid.set_src_addr(0x0a); +    BOOST_CHECK_EQUAL(sid.get(), 0x0a020304); +    BOOST_CHECK_EQUAL(sid.get_src_addr(), 0x0a); +    BOOST_CHECK_EQUAL(sid.get_src_endpoint(), 0x02); +    BOOST_CHECK_EQUAL(sid.get_dst_addr(), 0x03); +    BOOST_CHECK_EQUAL(sid.get_dst_endpoint(), 0x04); + +    sid.set_src_endpoint(0x0b); +    BOOST_CHECK_EQUAL(sid.get(), 0x0a0b0304); +    BOOST_CHECK_EQUAL(sid.get_src_addr(), 0x0a); +    BOOST_CHECK_EQUAL(sid.get_src_endpoint(), 0x0b); +    BOOST_CHECK_EQUAL(sid.get_dst_addr(), 0x03); +    BOOST_CHECK_EQUAL(sid.get_dst_endpoint(), 0x04); + +    sid.set_dst_addr(0x0c); +    BOOST_CHECK_EQUAL(sid.get(), 0x0a0b0c04); +    BOOST_CHECK_EQUAL(sid.get_src_addr(), 0x0a); +    BOOST_CHECK_EQUAL(sid.get_src_endpoint(), 0x0b); +    BOOST_CHECK_EQUAL(sid.get_dst_addr(), 0x0c); +    BOOST_CHECK_EQUAL(sid.get_dst_endpoint(), 0x04); + +    sid.set_dst_endpoint(0x0d); +    BOOST_CHECK_EQUAL(sid.get(), 0x0a0b0c0d); +    BOOST_CHECK_EQUAL(sid.get_src_addr(), 0x0a); +    BOOST_CHECK_EQUAL(sid.get_src_endpoint(), 0x0b); +    BOOST_CHECK_EQUAL(sid.get_dst_addr(), 0x0c); +    BOOST_CHECK_EQUAL(sid.get_dst_endpoint(), 0x0d); + +    sid.set_dst_xbarport(0xb); +    BOOST_CHECK_EQUAL(sid.get(), 0x0a0b0cbd); +    BOOST_CHECK_EQUAL(sid.get_src_addr(), 0x0a); +    BOOST_CHECK_EQUAL(sid.get_src_endpoint(), 0x0b); +    BOOST_CHECK_EQUAL(sid.get_dst_addr(), 0x0c); +    BOOST_CHECK_EQUAL(sid.get_dst_endpoint(), 0xbd); + +    sid.set_dst_blockport(0xc); +    BOOST_CHECK_EQUAL(sid.get(), 0x0a0b0cbc); +    BOOST_CHECK_EQUAL(sid.get_src_addr(), 0x0a); +    BOOST_CHECK_EQUAL(sid.get_src_endpoint(), 0x0b); +    BOOST_CHECK_EQUAL(sid.get_dst_addr(), 0x0c); +    BOOST_CHECK_EQUAL(sid.get_dst_endpoint(), 0xbc); + +    sid_t flipped_sid = sid.reversed(); +    BOOST_CHECK_EQUAL(flipped_sid.get(), 0x0cbc0a0b); + +    // In-place +    sid.reverse(); +    BOOST_CHECK_EQUAL(sid.get(), 0x0cbc0a0b); +} + +BOOST_AUTO_TEST_CASE(test_sid_t_from_str) { +    sid_t sid("1.2>3.4"); +    BOOST_CHECK_EQUAL(sid.get_src_addr(), 1); +    BOOST_CHECK_EQUAL(sid.get_src_endpoint(), 2); +    BOOST_CHECK_EQUAL(sid.get_dst_addr(), 3); +    BOOST_CHECK_EQUAL(sid.get_dst_endpoint(), 4); + +    sid = "01:02>03:10"; +    BOOST_CHECK_EQUAL(sid.get_src_addr(), 1); +    BOOST_CHECK_EQUAL(sid.get_src_endpoint(), 2); +    BOOST_CHECK_EQUAL(sid.get_dst_addr(), 3); +    BOOST_CHECK_EQUAL(sid.get_dst_endpoint(), 16); + +    sid = "01:06/03:10"; +    BOOST_CHECK_EQUAL(sid.get_src_addr(), 1); +    BOOST_CHECK_EQUAL(sid.get_src_endpoint(), 6); +    BOOST_CHECK_EQUAL(sid.get_dst_addr(), 3); +    BOOST_CHECK_EQUAL(sid.get_dst_endpoint(), 16); + +    sid = "01:02:04:10"; +    BOOST_CHECK_EQUAL(sid.get_src_addr(), 1); +    BOOST_CHECK_EQUAL(sid.get_src_endpoint(), 2); +    BOOST_CHECK_EQUAL(sid.get_dst_addr(), 4); +    BOOST_CHECK_EQUAL(sid.get_dst_endpoint(), 16); + +    BOOST_REQUIRE_THROW(sid_t fail_sid("foobar"), uhd::value_error); +    BOOST_REQUIRE_THROW(sid_t fail_sid("01:02:03:4"), uhd::value_error); +    BOOST_REQUIRE_THROW(sid_t fail_sid("01:02:03:004"), uhd::value_error); +    BOOST_REQUIRE_THROW(sid_t fail_sid("1.2.3.0004"), uhd::value_error); +} diff --git a/tools/uhd_txrx_debug_prints/uhd_txrx_debug_prints_README.md b/tools/uhd_txrx_debug_prints/uhd_txrx_debug_prints_README.md new file mode 100644 index 000000000..1425a4d96 --- /dev/null +++ b/tools/uhd_txrx_debug_prints/uhd_txrx_debug_prints_README.md @@ -0,0 +1,17 @@ +UHD TX/RX DEBUG PRINTS +====================== + +A tool for extensive debugging with UHD. + +Install +------- +Activate it by ticking `UHD_TXRX_DEBUG_PRINTS` in cmake-gui for your UHD installation. Then recompile and reinstall UHD. + +Use +--- +Run your application and pipe stderr to a file. +this is mostly done by <br> +`app_call 2> dbg_print_file.txt`<br> +After finishing the application offline processing of the gathered data is done with a python script called<br> +`uhd_txrx_debug_prints_graph.py`<br> +There are a lot of functions that help to preprocess your data and that describe the actual meaning of all the data points. in the end though, it comes down to the users needs what he wants to plot and see.
\ No newline at end of file diff --git a/tools/uhd_txrx_debug_prints/uhd_txrx_debug_prints_graph.py b/tools/uhd_txrx_debug_prints/uhd_txrx_debug_prints_graph.py new file mode 100755 index 000000000..b0f6681f6 --- /dev/null +++ b/tools/uhd_txrx_debug_prints/uhd_txrx_debug_prints_graph.py @@ -0,0 +1,415 @@ +#!/usr/bin/env python +# -*- coding: utf-8 -*- +# +# Copyright 2013-2014 Ettus Research LLC +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program.  If not, see <http://www.gnu.org/licenses/>. +# +""" +Plots the output of the streamers that is produced when DEBUG_TXRX +is enabled. +""" + +import argparse +import matplotlib.pyplot as plt +import numpy as np + +# This is a top level function to load a debug file. It will return a list of lists with all the data. +def get_data(filename): +    res = [] +    with open(filename) as f: +        for line in f.readlines(): +            line = line.rstrip('\n') +            s = line.split(',') +            res.append(s) +    res = fix_known_data_corruption_patterns(res) +    return res + +# Sometimes 'O' etc. get printed to stderr. This disturbs further processing. Thus, known pattern will be removed here. +def fix_known_data_corruption_patterns(data): +    # Some lines might be corrupted in the way that e.g. UHD prints sth else in the same line. +    # O, D, Press Enter... are known patterns. These should be fixed. +    counts = {'O': 0, 'D': 0, 'exit_msg': 0} +    for i in range(len(data)): +        if data[i][0].startswith('O'): +            counts['O'] += 1 +            data[i][0] = data[i][0].replace('O', '') +        if data[i][0].startswith('D'): +            counts['D'] += 1 +            data[i][0] = data[i][0].replace('D', '') +        if data[i][0].startswith('Press Enter to quit: '): +            counts['exit_msg'] += 1 +            data[i][0] = data[i][0].replace('Press Enter to quit: ', '') +    print counts +    return data + +# Extract lines with a certain prefix. +def extract_lines(prefix, data): +    res = [] +    for line in data: +        if line[0] == prefix: +            res.append(line[1:]) +    return res + +# boolean values are stored as true/false. Convert them to real booleans again. +def convert_to_bool(data): +    res = [] +    for d in data: +        res.append(d == "true") +    return res + +# The different outputs have different structures. With this function you can conert them back to their actual type. +def convert_data_set(data, reqtype): +    zdata = zip(*data) +    res = [] +    if len(zdata) == 0: +        return res +    for i in range(len(reqtype)): +        if reqtype[i] == np.bool: +            res.append(np.asarray(convert_to_bool(zdata[i]), dtype=np.bool)) +        else: +            res.append(np.asarray(zdata[i], dtype=reqtype[i])) +    return res + +# Wrapper for super_send_packet_handler data to be used with convert_data_set +def convert_super_send_packet_handler_data(data): +    # wallclock, timeout, avail_samps, sent_samps, sob, eob, has_time_spec, time_spec (ticks) +    reqtype = [np.uint64, np.float, np.int, np.int, np.bool, np.bool, np.bool, np.uint64] +    cdata = convert_data_set(data, reqtype) +    return cdata + +# same o' same o' +def convert_super_recv_packet_handler_data(data): +    # wallclock, timeout, requested_samps, received_samps, one_packet, error_code, sob, eob, more_fragments, fragment_offset, has_timespec, time_spec +    reqtype = [np.uint64, np.float, np.int, np.int, np.bool, np.int, np.bool, np.bool, np.bool, np.int, np.bool, np.uint64] +    cdata = convert_data_set(data, reqtype) +    return cdata + +def extract_super_recv_packet_handler_data(data): +    pref1 = "super_recv_packet_handler" +    pref2 = "recv" +    super_recv = extract_lines(pref1, data) +    recv = extract_lines(pref2, super_recv) +    recv = convert_super_recv_packet_handler_data(recv) +    return recv + +# Sometimes TX or RX is interrupted by system jiffies. Those are found by this function. +def find_jiffy(data, thr): +    res = [] +    last = data[0] +    for i in range(len(data)): +        if data[i] - last > thr: +            res.append([last, data[i]]) +        last = data[i] +    return res + + +# Get difference between tx and rx wallclock +def get_diff(tx, rx): +    print "get diff, len(rx) = ", len(rx) +    diff = [0] * len(rx[0]) +    for i in range(len(diff)): +        r = rx[3][i] +        idx = rx[0][i] - 1 # call count starts at 1. idx is 0 based. +        t = tx[3][idx] +        diff[i] = t - r +    return diff + + +def bps(samps, time): +    bp = [] +    last = time[0]-1000 +    for i in range(len(samps)): +        td = time[i] - last +        last = time[i] +        td = td /1e6 +        bp.append(samps[i] * 4 / td) +    return bp + + +# same as the other wrappers this time for libusb1 +def extract_libusb(trx, data): +    pr1 = "libusb1_zero_copy" +    pr2 = "libusb_async_cb" +    ldata = extract_lines(pr1, data) +    edata = extract_lines(pr2, ldata) +    d = extract_lines(trx, edata) +    # buff_num, actual_length, status, end_time, start_time +    reqtype = [np.int, np.int, np.int, np.uint64, np.uint64] +    return convert_data_set(d, reqtype) + + +# Extract data for stream buffers. Typically there are 16 TX and 16 RX buffers. And there numbers are static. Though the number of buffers might be changed and the constant parameters must be adjusted in this case. +def extract_txrx(data): +    tx = [[], [], [], [], []] +    rx = [[], [], [], [], []] +    for i in range(len(data[0])): +        print data[0][i] +        if data[0][i] > 31 and data[0][i] < 48: +            rx[0].append(data[0][i]) +            rx[1].append(data[1][i]) +            rx[2].append(data[2][i]) +            rx[3].append(data[3][i]) +            rx[4].append(data[4][i]) +            #print "tx\t", data[0][i], "\t", data[3][i], "\t", data[4][i] +        if data[0][i] > 47 and data[0][i] < 64: +            tx[0].append(data[0][i]) +            tx[1].append(data[1][i]) +            tx[2].append(data[2][i]) +            tx[3].append(data[3][i]) +            tx[4].append(data[4][i]) +            #print "rx\t", data[0][i], "\t", data[3][i], "\t", data[4][i] +    return [tx, rx] + +# Calculate momentary throughput +def throughput(data): +    start = data[2][0] +    total = data[1][-1] - data[2][0] +    print total +    thr = np.zeros(total) + +    for i in range(len(data[0])): +        s = data[2][i] - start +        f = data[1][i] - start +        ticks = data[1][i] - data[2][i] +        pertick = 1. * data[0][i] / ticks +        vals = [pertick] * ticks +        thr[s:f] = np.add(thr[s:f], vals) +        #print pertick +    print np.shape(thr) +    return thr + + +# Calculate a moving average +def ma(data, wl): +    ap = np.zeros(wl) +    data = np.concatenate((ap, data, ap)) +    print np.shape(data) +    res = np.zeros(len(data)-wl) +    for i in range(len(data)-wl): +        av = np.sum(data[i:i+wl]) / wl +        res[i] = av +        print i, "\t", av +    return res + + +def get_x_axis(stamps): +    scale = 10e-6 +    return np.multiply(stamps, scale) + + +# plot status codes. +def plot_status_codes_over_time(data, fignum): +    print "printing status numbers over time" + +    # extract and convert the data +    recv = extract_super_recv_packet_handler_data(data) +    libusb_rx = extract_libusb("rx", data) +    libusb_tx = extract_libusb("tx", data) + +    # Plot all data +    plt.figure(fignum) # Make sure these plots are printed to a new figure. +    #plt.plot(get_x_axis(libusb_rx[3]), libusb_rx[2], marker='x', label='RX') +    #plt.plot(get_x_axis(libusb_tx[3]), libusb_tx[2], marker='x', label='TX') + +    pos = 5 +    recv_error_codes = recv[pos] +    plt.plot(get_x_axis(recv[0]), recv_error_codes, marker='x', label='recv') +    plt.title("Status codes over timestamps") +    plt.ylabel("status codes") +    plt.xlabel("timestamps") +    plt.grid() +    plt.legend() + +    for i in range(len(recv[0])): +        if recv[pos][i] == 8: +            xaxis = get_x_axis([recv[0][i]]) +            plt.axvline(xaxis, color='b') + + +    # Get some statistics and print them too +    codes = [] +    code_dict = {} +    for i in range(len(recv_error_codes)): +        if recv_error_codes[i] != 0: +            code = rx_metadata_error_codes[recv_error_codes[i]] +            pair = [i, code] +            codes.append(pair) +            if not code_dict.has_key(code): +                code_dict[code] = 0 +            code_dict[code] += 1 +    print codes +    print code_dict + + +# plot rtt times as peaks. That's the fast and easy way. +def plot_rtt_times(data, fignum): +    print "plot RTT times" +    rx = extract_libusb("rx", data) +    tx = extract_libusb("tx", data) + +    scale = 10e-6 +    rx_diff = np.multiply(np.subtract(rx[3], rx[4]), scale) +    tx_diff = np.multiply(np.subtract(tx[3], tx[4]), scale) + +    plt.figure(fignum) +    plt.plot(get_x_axis(rx[3]), rx_diff, marker='x', ls='', label="rx RTT") +    plt.plot(get_x_axis(tx[3]), tx_diff, marker='x', ls='', label="tx RTT") +    plt.title("Round trip times") +    plt.ylabel("RTT (us)") +    plt.grid() +    plt.legend() + + +# plot RTT as actual lines as long as buffers are on the fly. +# This can take a long time if the function has to print a lot of small lines. Careful with this! +def plot_rtt_lines(data, fignum): +    print "plot RTT lines" +    rx = extract_libusb("rx", data) +    #tx = extract_libusb("tx", data) + +    if len(data) == 0 or len(rx) == 0: +        return + +    plt.figure(fignum) +    for i in range(len(rx[0])): +        if rx[0][i] > -1: +            start = rx[4][i] +            stop = rx[3][i] +            status = rx[2][i] + +            val = rx[0][i]#(stop - start) * scale +            if not status == 0: +                if status == 2: +                    print "status = ", status +                val +=0.5 +            xaxis = get_x_axis([start, stop]) +            plt.plot(xaxis, [val, val], marker='x') +    plt.ylabel('buffer number') + +    # Careful with these lines here. +    # Basically they should always have these values to separate CTRL, TX, RX buffer number blocks. +    # But these values can be adjusted. Thus, it requires you to adjust these lines +    plt.axhline(15.5) +    plt.axhline(31.5) +    plt.axhline(47.5) +    plt.grid() + + +# only plot on-the-fly buffers. +def plot_buff_otf(trx, nrange, data): +    d = extract_libusb(trx, data) +    res = [[], []] +    num = 0 +    for i in range(len(d[0])): +        if d[0][i] in nrange: +            res + + +# If there are still unknown lines after cleanup, they can be caught and printed here. +# This way you can check what got caught but shouldn't have been caught. +def get_unknown_lines(data): +    # These are the 3 known starting lines. More might be added in the future. +    known = ['super_recv_packet_handler', 'super_send_packet_handler', 'libusb1_zero_copy'] +    res = [] +    for i in range(len(data)): +        if not data[i][0] in known: +            print data[i] +            res.append(data[i]) +    return res + +# LUT for all the return codes +rx_metadata_error_codes = {0x0: "NONE", 0x1: "TIMEOUT", 0x2: "LATE_COMMAND", 0x4: "BROKEN_CHAIN", 0x8: "OVERFLOW", +                           0xc: "ALIGNMENT", 0xf: "BAD_PACKET"} + + +def parse_args(): +    """ Parse args, yo """ +    parser = argparse.ArgumentParser(description='Plot tool for debug prints.') +    parser.add_argument('-f', '--filename', default=None, help='Debug output file') +    return parser.parse_args() + +# Be really careful with the input files. They get pretty huge in a small time. +# Doing some of the plotting can eat up a lot of time then. +# Although this file contains a lot of functions by now, it is still left to the user to use everythin correctly. +def main(): +    args = parse_args() +    filename = args.filename +    print "get data from: ", filename +    #pref1 = "super_recv_packet_handler" +    #pref2 = "recv" + +    # Here you get the data from a file +    data = get_data(filename) +    #print "data len: ", len(data) + +    # extract lines with unknown content. +    unknown = get_unknown_lines(data) + +    # plot status codes and RTT lines. +    plot_status_codes_over_time(data, 0) +    plot_rtt_lines(data, 0) +    #plot_rtt_times(data, 0) + +    #[tx, rx] = extract_txrx(data) +    #print "txlen\t", len(tx[0]) +    #print "rxlen\t", len(rx[0]) + + +    #print "plot data" +    #plt.title(filename) +    #plt.plot(tx_data[0], tx_data[7], 'r', marker='o', label='TX') +    #plt.plot(rx_data[0], rx_data[11], 'g', marker='x', label='RX') +    #for j in jiffies: +    #    plt.axvline(x=j[0], color='r') +    #    plt.axvline(x=j[1], color='g') +    # +    #plt.xlabel('wallclock (us)') +    #plt.ylabel('timestamp (ticks)') +    #plt.legend() + +    #lencal = 200000 + +    # calculate and plot rx throughput +    #rxthr = throughput([rx[1], rx[3], rx[4]]) +    #plt.plot(rxthr[0:lencal]) +    #rxav = ma(rxthr[0:lencal], 2000) +    #plt.plot(rxav) + +    # calculate and plot tx throughput +    #txthr = throughput([tx[1], tx[3], tx[4]]) +    #plt.plot(txthr[0:lencal]) +    #txav = ma(txthr[0:lencal], 20) +    #plt.plot(txav) + + +    #bp = bps(data[1], data[2]) +    #print np.sum(bp) +    #ave = np.sum(bp)/len(bp) +    #print ave +    #plt.plot(tx_data[0], tx_data[2], 'r') +    #plt.plot(rx_data[0], rx_data[2], 'b') +    #plt.plot(bp, 'g', marker='+') +    #plt.plot(np.multiply(data[1], 1e6), 'r', marker='o') +    #plt.plot(np.multiply(data[3],1e2), 'b', marker='x') +    #plt.plot(ave) + +    # in the end, put a grid on the graph and show it. +    plt.grid() +    plt.show() + +if __name__ == '__main__': +    print "[WARNING] This tool is in alpha status. Only use if you know what you're doing!" +    main() + | 
