diff options
Diffstat (limited to 'host')
36 files changed, 785 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); +} | 
