Compare commits
	
		
			40 Commits
		
	
	
		
			ethstream-
			...
			ethstream-
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
|   | 5a14055029 | ||
|   | 25bed4b644 | ||
|   | cd63e2fb11 | ||
|   | 329871bf56 | ||
|   | f4f809eee3 | ||
|   | ec1ae04a5b | ||
|   | 67163dc1a2 | ||
|   | a29bf180ab | ||
|   | 4cccb783f0 | ||
|   | 5e7f4ac97e | ||
|   | a57c0c22c2 | ||
|   | 2e3f2f4b23 | ||
|   | 4592b6a75b | ||
|   | d1447e2af8 | ||
|   | b9130788f2 | ||
|   | ed150ead35 | ||
|   | 519a0c2e29 | ||
|   | 190fd3f3a7 | ||
|   | 9c174d12b0 | ||
|   | b1e049eed0 | ||
|   | d461365275 | ||
|   | a4eede145b | ||
|   | 6562c0b787 | ||
|   | 7f79ec8ff3 | ||
|   | 1dd09ea52d | ||
|   | ded6c7e5f4 | ||
|   | f2d6566051 | ||
|   | 357e808986 | ||
|   | 0ed2935deb | ||
|   | 1a2ed51d0c | ||
|   | 679b4ef64f | ||
|   | 770a83b8f7 | ||
|   | 627a1bf22b | ||
|   | 43c8e19c67 | ||
|   | 4aab606162 | ||
|   | c81d372d23 | ||
|   | 407a91c764 | ||
|   | 1d760e033b | ||
|   | 57cfac2bec | ||
|   | 5b731f3e7e | 
							
								
								
									
										25
									
								
								Makefile
									
									
									
									
									
								
							
							
						
						
									
										25
									
								
								Makefile
									
									
									
									
									
								
							| @@ -19,9 +19,10 @@ PREFIX = /usr/local | ||||
| MANPATH = ${PREFIX}/man/man1 | ||||
| BINPATH = ${PREFIX}/bin | ||||
|  | ||||
| WINCC = i386-mingw32-gcc | ||||
| #WINCC = i386-mingw32-gcc | ||||
| WINCC = i586-mingw32msvc-gcc | ||||
| WINCFLAGS += $(CFLAGS) | ||||
| WINLDFLAGS += $(LDFLAGS) -lws2_32 -s | ||||
| WINLDFLAGS += $(LDFLAGS) -lws2_32 -liphlpapi -s | ||||
|  | ||||
| # Targets | ||||
|  | ||||
| @@ -32,11 +33,10 @@ default: lin | ||||
| all: lin win | ||||
|  | ||||
| .PHONY: lin | ||||
| lin: ljtest ethstream ljconfig \ | ||||
| 	ethstream.1 ljconfig.1 | ||||
| lin: ethstream ethstream.1 ethstream.txt | ||||
|  | ||||
| .PHONY: win | ||||
| win: ljtest.exe ethstream.exe ljconfig.exe | ||||
| win: ethstream.exe | ||||
|  | ||||
|  | ||||
| version.h: VERSION | ||||
| @@ -46,17 +46,11 @@ version.h: VERSION | ||||
| # Object files for each executable | ||||
|  | ||||
| obj-common = opt.o ue9.o ue9error.o netutil.o debug.o nerdjack.o | ||||
| obj-ljconfig = ljconfig.o $(obj-common) | ||||
| obj-ethstream = ethstream.o $(obj-common) | ||||
| obj-ljtest = ljtest.o $(obj-common) | ||||
|  | ||||
| ljconfig: $(obj-ljconfig) | ||||
| ethstream: $(obj-ethstream) | ||||
| ljtest: $(obj-ljtest) | ||||
|  | ||||
| ljconfig.exe: $(obj-ljconfig:.o=.obj) compat-win32.obj | ||||
| ethstream.exe: $(obj-ethstream:.o=.obj) compat-win32.obj | ||||
| ljtest.exe: $(obj-ljtest:.o=.obj) compat-win32.obj | ||||
|  | ||||
| # Manpages | ||||
|  | ||||
| @@ -92,16 +86,19 @@ dist: version.h | ||||
|  | ||||
| .PHONY: clean distclean | ||||
| clean distclean: | ||||
| 	rm -f *.o *.obj *.exe ethstream ljtest ljconfig core *.d *.1 *.txt | ||||
| 	rm -f *.o *.obj *.exe ethstream core *.d *.dobj *.1 *.txt | ||||
|  | ||||
| # Dependency tracking: | ||||
|  | ||||
| allsources = $(wildcard *.c) | ||||
|  | ||||
| -include $(allsources:.c=.d) | ||||
| %.o : %.c | ||||
| 	$(COMPILE.c) -MP -MMD -MT '$*.obj' -o $@ $< | ||||
| 	$(COMPILE.c) -MP -MMD -MT '$*.o' -MF '$*.d' -o $@ $< | ||||
|  | ||||
| -include $(allsources:.c=.dobj) | ||||
| %.obj : %.c | ||||
| 	$(WINCC) $(WINCFLAGS) -MP -MMD -MT '$*.o' -c -o $@ $< | ||||
| 	$(WINCC) $(WINCFLAGS) -MP -MMD -MT '$*.obj' -MF '$*.dobj' -c -o $@ $< | ||||
|  | ||||
| # Win32 executable | ||||
|  | ||||
|   | ||||
							
								
								
									
										7
									
								
								README
									
									
									
									
									
								
							
							
						
						
									
										7
									
								
								README
									
									
									
									
									
								
							| @@ -2,5 +2,10 @@ Labjack/Nerdjack Tools | ||||
| by Jim Paris <jim@jtan.com> | ||||
| with modifications by Zach Clifford <zacharyc@mit.edu> | ||||
|  | ||||
| These tools are for interacting with the LabJack UE9 or the NerdJack over the Ethernet interface.  More information about the UE9 device: | ||||
| These tools are for interacting with the LabJack UE9 or the NerdJack | ||||
| over the Ethernet interface.  More information about the UE9 device: | ||||
|   http://www.labjack.com/labjack_ue9.php | ||||
|  | ||||
| The NerdJack device is a custom board made in LEES by Zach Clifford. | ||||
|  | ||||
| Use ethstream -h or ethstream -X for usage instructions and examples. | ||||
|   | ||||
| @@ -14,36 +14,37 @@ static struct { | ||||
| 	char *msg; | ||||
| } win32_error[] = { | ||||
| 	/* Errors that we might vaguely expect to see */ | ||||
| 	{ WSAEINTR, "Winsock: Interrupted system call" }, | ||||
| 	{ WSAEBADF, "Winsock: Bad file number" }, | ||||
| 	{ WSAEFAULT, "Winsock: Bad address" }, | ||||
| 	{ WSAEINVAL, "Winsock: Invalid argument" }, | ||||
| 	{ WSAEMFILE, "Winsock: Too many open files" }, | ||||
| 	{ WSAEWOULDBLOCK, "Winsock: Operation would block" }, | ||||
| 	{ WSAEINPROGRESS, "Winsock: Operation now in progress" }, | ||||
| 	{ WSAEALREADY, "Winsock: Operation already in progress" }, | ||||
| 	{ WSAENOTSOCK, "Winsock: Socket operation on nonsocket" }, | ||||
| 	{ WSAEADDRINUSE, "Winsock: Address already in use" }, | ||||
| 	{ WSAEADDRNOTAVAIL, "Winsock: Cannot assign requested address" }, | ||||
| 	{ WSAENETDOWN, "Winsock: Network is down" }, | ||||
| 	{ WSAENETUNREACH, "Winsock: Network is unreachable" }, | ||||
| 	{ WSAENETRESET, "Winsock: Network dropped connection on reset" }, | ||||
| 	{ WSAECONNABORTED, "Winsock: Software caused connection abort" }, | ||||
| 	{ WSAECONNRESET, "Winsock: Connection reset by peer" }, | ||||
| 	{ WSAETIMEDOUT, "Winsock: Connection timed out" }, | ||||
| 	{ WSAECONNREFUSED, "Winsock: Connection refused" }, | ||||
| 	{ WSAEHOSTDOWN, "Winsock: Host is down" }, | ||||
| 	{ WSAEHOSTUNREACH, "Winsock: No route to host" }, | ||||
| 	{ WSAVERNOTSUPPORTED, "Winsock: Unsupported Winsock version" }, | ||||
| 	{ ETIMEDOUT, "Connection timed out" }, | ||||
| 	{ ENOTCONN, "Not connected" }, | ||||
| 	{ -1, NULL }, | ||||
| 	{ WSAEINTR, "Winsock: Interrupted system call"}, | ||||
| 	{ WSAEBADF, "Winsock: Bad file number"}, | ||||
| 	{ WSAEFAULT, "Winsock: Bad address"}, | ||||
| 	{ WSAEINVAL, "Winsock: Invalid argument"}, | ||||
| 	{ WSAEMFILE, "Winsock: Too many open files"}, | ||||
| 	{ WSAEWOULDBLOCK, "Winsock: Operation would block"}, | ||||
| 	{ WSAEINPROGRESS, "Winsock: Operation now in progress"}, | ||||
| 	{ WSAEALREADY, "Winsock: Operation already in progress"}, | ||||
| 	{ WSAENOTSOCK, "Winsock: Socket operation on nonsocket"}, | ||||
| 	{ WSAEADDRINUSE, "Winsock: Address already in use"}, | ||||
| 	{ WSAEADDRNOTAVAIL, "Winsock: Cannot assign requested address"}, | ||||
| 	{ WSAENETDOWN, "Winsock: Network is down"}, | ||||
| 	{ WSAENETUNREACH, "Winsock: Network is unreachable"}, | ||||
| 	{ WSAENETRESET, "Winsock: Network dropped connection on reset"}, | ||||
| 	{ WSAECONNABORTED, "Winsock: Software caused connection abort"}, | ||||
| 	{ WSAECONNRESET, "Winsock: Connection reset by peer"}, | ||||
| 	{ WSAETIMEDOUT, "Winsock: Connection timed out"}, | ||||
| 	{ WSAECONNREFUSED, "Winsock: Connection refused"}, | ||||
| 	{ WSAEHOSTDOWN, "Winsock: Host is down"}, | ||||
| 	{ WSAEHOSTUNREACH, "Winsock: No route to host"}, | ||||
| 	{ WSAVERNOTSUPPORTED, "Winsock: Unsupported Winsock version"}, | ||||
| 	{ ETIMEDOUT, "Connection timed out"}, | ||||
| 	{ ENOTCONN, "Not connected"}, | ||||
| 	{ -1, NULL}, | ||||
| }; | ||||
|  | ||||
| char *compat_strerror(int errnum) | ||||
| { | ||||
| 	int i; | ||||
| 	static char buf[128]; | ||||
| 	 | ||||
|  | ||||
| 	for (i = 0; win32_error[i].num != -1; i++) | ||||
| 		if (errnum == win32_error[i].num) | ||||
| 			return win32_error[i].msg; | ||||
| @@ -80,4 +81,3 @@ char *compat_strerror(int errnum) | ||||
| } | ||||
| */ | ||||
| #endif | ||||
|  | ||||
|   | ||||
							
								
								
									
										3
									
								
								debug.c
									
									
									
									
									
								
							
							
						
						
									
										3
									
								
								debug.c
									
									
									
									
									
								
							| @@ -4,7 +4,7 @@ | ||||
|  | ||||
| int verb_count = 0; | ||||
|  | ||||
| int func_fprintf(const char *func, FILE *stream, const char *format, ...) | ||||
| int func_fprintf(const char *func, FILE * stream, const char *format, ...) | ||||
| { | ||||
| 	va_list ap; | ||||
| 	int ret; | ||||
| @@ -15,4 +15,3 @@ int func_fprintf(const char *func, FILE *stream, const char *format, ...) | ||||
| 	va_end(ap); | ||||
| 	return ret; | ||||
| } | ||||
|  | ||||
|   | ||||
							
								
								
									
										4
									
								
								debug.h
									
									
									
									
									
								
							
							
						
						
									
										4
									
								
								debug.h
									
									
									
									
									
								
							| @@ -14,8 +14,8 @@ extern int verb_count; | ||||
|  | ||||
| #include <stdio.h> | ||||
|  | ||||
| int func_fprintf(const char *func, FILE *stream, const char *format,  | ||||
| 		 ...) __attribute__ ((format (printf, 3, 4))); | ||||
| int func_fprintf(const char *func, FILE * stream, const char *format, | ||||
| 		 ...) __attribute__ ((format(printf, 3, 4))); | ||||
|  | ||||
| #define debug(x...) ({ \ | ||||
| 	if(verb_count >= 2) \ | ||||
|   | ||||
							
								
								
									
										589
									
								
								ethstream.c
									
									
									
									
									
								
							
							
						
						
									
										589
									
								
								ethstream.c
									
									
									
									
									
								
							| @@ -7,11 +7,6 @@ | ||||
|  * License as published by the Free Software Foundation; see COPYING. | ||||
|  */ | ||||
|  | ||||
| /* ljstream: Stream data from the first N (1-14) analog inputs. | ||||
|    Resolution is set to 12-bit and all channels are in bipolar (-5 to | ||||
|    +5V) mode. | ||||
| */ | ||||
|  | ||||
| #include <stdint.h> | ||||
| #include <stdlib.h> | ||||
| #include <stdio.h> | ||||
| @@ -29,40 +24,58 @@ | ||||
| #include "opt.h" | ||||
| #include "version.h" | ||||
| #include "compat.h" | ||||
| #include "ethstream.h" | ||||
|  | ||||
| #include "example.inc" | ||||
|  | ||||
| #define DEFAULT_HOST "192.168.1.209" | ||||
| #define UE9_COMMAND_PORT 52360 | ||||
| #define UE9_DATA_PORT 52361 | ||||
|  | ||||
| struct callbackInfo {  | ||||
| #define MAX_CHANNELS 256 | ||||
|  | ||||
| struct callbackInfo { | ||||
| 	struct ue9Calibration calib; | ||||
| 	int convert; | ||||
| 	int maxlines; | ||||
| }; | ||||
|  | ||||
| struct options opt[] = { | ||||
| 	{ 'a', "address", "string", "host/address of UE9 (192.168.1.209)" }, | ||||
| 	{ 'n', "numchannels", "n", "sample the first N ADC channels (2)" }, | ||||
|     { 'N', "nerdjack", NULL, "Use NerdJack device instead" }, | ||||
|     { 'd', "detect", NULL, "Detect NerdJack IP address" }, | ||||
|     { 'p', "precision", "0-3", "Set precision on NerdJack (0 - max range, 1 - max precision)"},  | ||||
| 	{ 'C', "channels", "a,b,c", "sample channels a, b, and c" }, | ||||
| 	{ 'r', "rate", "hz", "sample each channel at this rate (8000.0)" }, | ||||
| 	{ 'o', "oneshot", NULL, "don't retry in case of errors" }, | ||||
| 	{ 'f', "forceretry", NULL, "retry no matter what happens" }, | ||||
| 	{ 'c', "convert", NULL, "display output in volts" }, | ||||
| 	{ 'l', "lines", "num", "if set, output this many lines and quit" }, | ||||
| 	{ 'h', "help", NULL, "this help" }, | ||||
| 	{ 'v', "verbose", NULL, "be verbose" }, | ||||
| 	{ 'V', "version", NULL, "show version number and exit" }, | ||||
| 	{ 0, NULL, NULL, NULL } | ||||
| 	{'a', "address", "string", "host/address of device (192.168.1.209)"}, | ||||
| 	{'n', "numchannels", "n", "sample the first N ADC channels (2)"}, | ||||
| 	{'C', "channels", "a,b,c", "sample channels a, b, and c"}, | ||||
| 	{'r', "rate", "hz", "sample each channel at this rate (8000.0)"}, | ||||
|  | ||||
| 	{'L', "labjack", NULL, "Force LabJack device"}, | ||||
| 	{'t', "timers", "a,b,c", "set LabJack timer modes to a, b, and c"}, | ||||
| 	{'T', "timerdivisor", "n", "set LabJack timer divisor to n"}, | ||||
|  | ||||
| 	{'N', "nerdjack", NULL, "Force NerdJack device"}, | ||||
| 	{'d', "detect", NULL, "Detect NerdJack IP address"}, | ||||
| 	{'R', "range", "a,b", | ||||
| 	 "Set range on NerdJack for channels 0-5,6-11 to either 5 or 10 (10,10)"}, | ||||
| 	{'o', "oneshot", NULL, "don't retry in case of errors"}, | ||||
| 	{'f', "forceretry", NULL, "retry no matter what happens"}, | ||||
| 	{'c', "convert", NULL, "convert output to volts"}, | ||||
| 	{'H', "converthex", NULL, "convert output to hex"}, | ||||
| 	{'m', "showmem", NULL, "output memory stats with data (NJ only)"}, | ||||
| 	{'l', "lines", "num", "if set, output this many lines and quit"}, | ||||
| 	{'h', "help", NULL, "this help"}, | ||||
| 	{'v', "verbose", NULL, "be verbose"}, | ||||
| 	{'V', "version", NULL, "show version number and exit"}, | ||||
| 	{'i', "info", NULL, "get info from device (NJ only)"}, | ||||
| 	{'X', "examples", NULL, "show ethstream examples and exit"}, | ||||
| 	{0, NULL, NULL, NULL} | ||||
| }; | ||||
|  | ||||
| int doStream(const char *address, uint8_t scanconfig, uint16_t scaninterval, | ||||
| 	     int *channel_list, int channel_count, int convert, int maxlines); | ||||
| int nerdDoStream(const char *address, int *channel_list, int channel_count, int precision,  | ||||
|             unsigned short period, int convert, int lines); | ||||
| int data_callback(int channels, uint16_t *data, void *context); | ||||
| 	     int *channel_list, int channel_count,  | ||||
| 	     int *timer_mode_list, int timer_mode_count, int timer_divisor, | ||||
| 	     int convert, int maxlines); | ||||
| int nerdDoStream(const char *address, int *channel_list, int channel_count, | ||||
| 		 int precision, unsigned long period, int convert, int lines, | ||||
| 		 int showmem); | ||||
| int data_callback(int channels, uint16_t * data, void *context); | ||||
|  | ||||
| int columns_left = 0; | ||||
| void handle_sig(int sig) | ||||
| @@ -87,19 +100,23 @@ int main(int argc, char *argv[]) | ||||
| 	double actual_rate; | ||||
| 	int oneshot = 0; | ||||
| 	int forceretry = 0; | ||||
| 	int convert = 0; | ||||
| 	int convert = CONVERT_DEC; | ||||
| 	int showmem = 0; | ||||
| 	int inform = 0; | ||||
| 	uint8_t scanconfig; | ||||
| 	uint16_t scaninterval; | ||||
| #if UE9_CHANNELS > NERDJACK_CHANNELS | ||||
| 	int channel_list[UE9_CHANNELS]; | ||||
| #else | ||||
|     int channel_list[NERDJACK_CHANNELS]; | ||||
| #endif | ||||
| 	int timer_mode_list[UE9_TIMERS]; | ||||
| 	int timer_mode_count = 0; | ||||
| 	int timer_divisor = 1; | ||||
| 	int channel_list[MAX_CHANNELS]; | ||||
| 	int channel_count = 0; | ||||
|     int nerdjack = 0; | ||||
|     int detect = 0; | ||||
|     int precision = 0; | ||||
|     int period = NERDJACK_CLOCK_RATE / desired_rate; | ||||
| 	int nerdjack = 0; | ||||
| 	int labjack = 0; | ||||
| 	int detect = 0; | ||||
| 	int precision = 0; | ||||
| 	int addressSpecified = 0; | ||||
| 	int donerdjack = 0; | ||||
| 	unsigned long period = NERDJACK_CLOCK_RATE / desired_rate; | ||||
|  | ||||
| 	/* Parse arguments */ | ||||
| 	opt_init(&optind); | ||||
| @@ -108,11 +125,12 @@ int main(int argc, char *argv[]) | ||||
| 		case 'a': | ||||
| 			free(address); | ||||
| 			address = strdup(optarg); | ||||
| 			addressSpecified = 1; | ||||
| 			break; | ||||
| 		case 'n': | ||||
| 			channel_count = 0; | ||||
| 			tmp = strtol(optarg, &endp, 0); | ||||
| 			if (*endp || tmp < 1 || tmp > UE9_CHANNELS) { | ||||
| 			if (*endp || tmp < 1 || tmp > MAX_CHANNELS) { | ||||
| 				info("bad number of channels: %s\n", optarg); | ||||
| 				goto printhelp; | ||||
| 			} | ||||
| @@ -123,29 +141,51 @@ int main(int argc, char *argv[]) | ||||
| 			channel_count = 0; | ||||
| 			do { | ||||
| 				tmp = strtol(optarg, &endp, 0); | ||||
| 				if (*endp != '\0' && *endp != ',')  { | ||||
|                 //|| tmp < 0 || tmp >= UE9_CHANNELS) { | ||||
| 					info("bad channel number: %s\n", optarg); | ||||
| 				if (*endp != '\0' && *endp != ',') { | ||||
| 					info("bad channel number: %s\n", | ||||
| 					     optarg); | ||||
| 					goto printhelp; | ||||
| 				} | ||||
|                 //We do not want to overflow channel_list, so we need the check here | ||||
|                 //The rest of the sanity checking can come later after we know whether this is a  | ||||
|                 //LabJack or a NerdJack | ||||
| #if UE9_CHANNELS > NERDJACK_CHANNELS | ||||
| 				if (channel_count >= UE9_CHANNELS) { | ||||
| #else | ||||
|                 if (channel_count >= NERDJACK_CHANNELS) { | ||||
| #endif | ||||
|                 	info("error: too many channels specified\n"); | ||||
| 				//We do not want to overflow channel_list, so we need the check here | ||||
| 				//The rest of the sanity checking can come later after we know | ||||
| 				//whether this is a  | ||||
| 				//LabJack or a NerdJack | ||||
| 				if (channel_count >= MAX_CHANNELS) { | ||||
| 					info("error: too many channels specified\n"); | ||||
| 					goto printhelp; | ||||
| 				} | ||||
| 				channel_list[channel_count++] = tmp; | ||||
| 				optarg = endp + 1; | ||||
| 			} while (*endp); | ||||
| 			break;			 | ||||
| 			} | ||||
| 			while (*endp); | ||||
| 			break; | ||||
| 		case 't':	/* labjack only */ | ||||
| 			timer_mode_count = 0; | ||||
| 			do { | ||||
| 				tmp = strtol(optarg, &endp, 0); | ||||
| 				if (*endp != '\0' && *endp != ',') { | ||||
| 					info("bad timer mode: %s\n", optarg); | ||||
| 					goto printhelp; | ||||
| 				} | ||||
| 				if (timer_mode_count >= UE9_TIMERS) { | ||||
| 					info("error: too many timers specified\n"); | ||||
| 					goto printhelp; | ||||
| 				} | ||||
| 				timer_mode_list[timer_mode_count++] = tmp; | ||||
| 				optarg = endp + 1; | ||||
| 			} | ||||
| 			while (*endp); | ||||
| 			break; | ||||
| 		case 'T':	/* labjack only */ | ||||
| 			timer_divisor = strtod(optarg, &endp); | ||||
| 			if (*endp || timer_divisor < 0 || timer_divisor > 255) { | ||||
| 				info("bad timer divisor: %s\n", optarg); | ||||
| 				goto printhelp; | ||||
| 			} | ||||
| 			break; | ||||
| 		case 'r': | ||||
| 			desired_rate = strtod(optarg, &endp); | ||||
| 			if(*endp || desired_rate <= 0) { | ||||
| 			if (*endp || desired_rate <= 0) { | ||||
| 				info("bad rate: %s\n", optarg); | ||||
| 				goto printhelp; | ||||
| 			} | ||||
| @@ -157,38 +197,89 @@ int main(int argc, char *argv[]) | ||||
| 				goto printhelp; | ||||
| 			} | ||||
| 			break; | ||||
|         case 'p': | ||||
|             precision++; | ||||
|             break; | ||||
|         case 'N': | ||||
|             nerdjack++; | ||||
|             break; | ||||
|         case 'd': | ||||
|             detect++; | ||||
|             break; | ||||
| 		case 'R': | ||||
| 			tmp = strtol(optarg, &endp, 0); | ||||
| 			if (*endp != ',') { | ||||
| 				info("bad range number: %s\n", optarg); | ||||
| 				goto printhelp; | ||||
| 			} | ||||
| 			if (tmp != 5 && tmp != 10) { | ||||
| 				info("valid choices for range are 5 or 10\n"); | ||||
| 				goto printhelp; | ||||
| 			} | ||||
| 			if (tmp == 5) | ||||
| 				precision = precision + 1; | ||||
|  | ||||
| 			optarg = endp + 1; | ||||
| 			if (*endp == '\0') { | ||||
| 				info("Range needs two numbers, one for channels 0-5 and another for 6-11\n"); | ||||
| 				goto printhelp; | ||||
| 			} | ||||
| 			tmp = strtol(optarg, &endp, 0); | ||||
| 			if (*endp != '\0') { | ||||
| 				info("Range needs only two numbers, one for channels 0-5 and another for 6-11\n"); | ||||
| 				goto printhelp; | ||||
| 			} | ||||
| 			if (tmp != 5 && tmp != 10) { | ||||
| 				info("valid choices for range are 5 or 10\n"); | ||||
| 				goto printhelp; | ||||
| 			} | ||||
| 			if (tmp == 5) | ||||
| 				precision = precision + 2; | ||||
| 			break; | ||||
| 		case 'N': | ||||
| 			nerdjack++; | ||||
| 			break; | ||||
| 		case 'L': | ||||
| 			labjack++; | ||||
| 			break; | ||||
| 		case 'd': | ||||
| 			detect++; | ||||
| 			break; | ||||
| 		case 'o': | ||||
| 			oneshot++; | ||||
| 			break; | ||||
| 		case 'f': | ||||
| 		  	forceretry++; | ||||
| 			forceretry++; | ||||
| 			break; | ||||
| 		case 'c': | ||||
| 			convert++; | ||||
| 			if (convert != 0) { | ||||
| 				info("specify only one conversion type\n"); | ||||
| 				goto printhelp; | ||||
| 			} | ||||
| 			convert = CONVERT_VOLTS; | ||||
| 			break; | ||||
| 		case 'H': | ||||
| 			if (convert != 0) { | ||||
| 				info("specify only one conversion type\n"); | ||||
| 				goto printhelp; | ||||
| 			} | ||||
| 			convert = CONVERT_HEX; | ||||
| 			break; | ||||
| 		case 'm': | ||||
| 			showmem++; | ||||
| 		case 'v': | ||||
| 			verb_count++; | ||||
| 			break; | ||||
| 		case 'X': | ||||
| 			printf("%s", examplestring); | ||||
| 			return 0; | ||||
| 			break; | ||||
| 		case 'V': | ||||
| 			printf("ljstream " VERSION "\n"); | ||||
| 			printf("etherstream " VERSION "\n"); | ||||
| 			printf("Written by Jim Paris <jim@jtan.com>\n"); | ||||
| 			printf("and Zachary Clifford <zacharyc@mit.edu>.\n"); | ||||
| 			printf("This program comes with no warranty and is " | ||||
| 			       "provided under the GPLv2.\n"); | ||||
| 			return 0; | ||||
| 			break; | ||||
| 		case 'i': | ||||
| 			inform++; | ||||
| 			break; | ||||
| 		case 'h': | ||||
| 			help = stdout; | ||||
| 		default: | ||||
| 		printhelp: | ||||
|  printhelp: | ||||
| 			fprintf(help, "Usage: %s [options]\n", *argv); | ||||
| 			opt_help(opt, help); | ||||
| 			fprintf(help, "Read data from the specified Labjack UE9" | ||||
| @@ -196,32 +287,93 @@ int main(int argc, char *argv[]) | ||||
| 			return (help == stdout) ? 0 : 1; | ||||
| 		} | ||||
| 	} | ||||
|      | ||||
|     if (nerdjack) { | ||||
|         if (channel_count > NERDJACK_CHANNELS) { | ||||
|             info("Too many channels for NerdJack\n"); | ||||
|             goto printhelp; | ||||
|         } | ||||
|         for (i = 0; i < channel_count; i++) { | ||||
|             if (channel_list[i] >= NERDJACK_CHANNELS) { | ||||
|                 info("Channel is out of NerdJack range: %d\n",channel_list[i]); | ||||
|                 goto printhelp; | ||||
|             } | ||||
|         } | ||||
|     } else { | ||||
|         if (channel_count > UE9_CHANNELS) { | ||||
|             info("Too many channels for LabJack\n"); | ||||
|             goto printhelp; | ||||
|         } | ||||
|         for (i = 0; i < channel_count; i++) { | ||||
|             if (channel_list[i] >= UE9_CHANNELS) { | ||||
|                 info("Channel is out of LabJack range: %d\n",channel_list[i]); | ||||
|                 goto printhelp; | ||||
|             } | ||||
|         } | ||||
|     } | ||||
|          | ||||
|          | ||||
|  | ||||
| 	if (detect && labjack) { | ||||
| 		info("The LabJack does not support autodetection\n"); | ||||
| 		goto printhelp; | ||||
| 	} | ||||
|  | ||||
| 	if (detect && !nerdjack) { | ||||
| 		info("Only the NerdJack supports autodetection - assuming -N option\n"); | ||||
| 		nerdjack = 1; | ||||
| 	} | ||||
|  | ||||
| 	if (detect && addressSpecified) { | ||||
| 		info("Autodetection and specifying address are mutually exclusive\n"); | ||||
| 		goto printhelp; | ||||
| 	} | ||||
|  | ||||
| 	if (nerdjack && labjack) { | ||||
| 		info("Nerdjack and Labjack options are mutually exclusive\n"); | ||||
| 		goto printhelp; | ||||
| 	} | ||||
|  | ||||
| 	donerdjack = nerdjack; | ||||
|  | ||||
| 	//First if no options were supplied try the Nerdjack | ||||
| 	//The second time through, donerdjack will be true and this will not fire | ||||
| 	if (!nerdjack && !labjack) { | ||||
| 		info("No device specified...Defaulting to Nerdjack\n"); | ||||
| 		donerdjack = 1; | ||||
| 	} | ||||
|  | ||||
|  doneparse: | ||||
|  | ||||
| 	if (inform) { | ||||
| 		//We just want information from NerdJack | ||||
| 		if (!detect) { | ||||
| 			if (nerd_get_version(address) < 0) { | ||||
| 				info("Could not find NerdJack at specified address\n"); | ||||
| 			} else { | ||||
| 				return 0; | ||||
| 			} | ||||
| 		} | ||||
| 		info("Autodetecting NerdJack address\n"); | ||||
| 		free(address); | ||||
| 		if (nerdjack_detect(address) < 0) { | ||||
| 			info("Error with autodetection\n"); | ||||
| 			goto printhelp; | ||||
| 		} else { | ||||
| 			info("Found NerdJack at address: %s\n", address); | ||||
| 			if (nerd_get_version(address) < 0) { | ||||
| 				info("Error getting NerdJack version\n"); | ||||
| 				goto printhelp; | ||||
| 			} | ||||
| 			return 0; | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	if (donerdjack) { | ||||
| 		if (channel_count > NERDJACK_CHANNELS) { | ||||
| 			info("Too many channels for NerdJack\n"); | ||||
| 			goto printhelp; | ||||
| 		} | ||||
| 		for (i = 0; i < channel_count; i++) { | ||||
| 			if (channel_list[i] >= NERDJACK_CHANNELS) { | ||||
| 				info("Channel is out of NerdJack range: %d\n", | ||||
| 				     channel_list[i]); | ||||
| 				goto printhelp; | ||||
| 			} | ||||
| 		} | ||||
| 	} else { | ||||
| 		if (channel_count > UE9_MAX_CHANNEL_COUNT) { | ||||
| 			info("Too many channels for LabJack\n"); | ||||
| 			goto printhelp; | ||||
| 		} | ||||
| 		for (i = 0; i < channel_count; i++) { | ||||
| 			if (channel_list[i] > UE9_MAX_CHANNEL) { | ||||
| 				info("Channel is out of LabJack range: %d\n", | ||||
| 				     channel_list[i]); | ||||
| 				goto printhelp; | ||||
| 			} | ||||
| 		} | ||||
| 	} | ||||
| 	 | ||||
| 	/* Timer requires Labjack */ | ||||
| 	if (timer_mode_count && !labjack) { | ||||
| 		info("Can't use timers on NerdJack\n"); | ||||
| 		goto printhelp; | ||||
| 	} | ||||
|  | ||||
| 	if (optind < argc) { | ||||
| 		info("error: too many arguments (%s)\n\n", argv[optind]); | ||||
| @@ -232,7 +384,7 @@ int main(int argc, char *argv[]) | ||||
| 		info("forceretry and oneshot options are mutually exclusive\n"); | ||||
| 		goto printhelp; | ||||
| 	} | ||||
| 	 | ||||
|  | ||||
| 	/* Two channels if none specified */ | ||||
| 	if (channel_count == 0) { | ||||
| 		channel_list[channel_count++] = 0; | ||||
| @@ -247,24 +399,22 @@ int main(int argc, char *argv[]) | ||||
| 	} | ||||
|  | ||||
| 	/* Figure out actual rate. */ | ||||
|     if (nerdjack) { | ||||
|         if (nerdjack_choose_scan(desired_rate, &actual_rate, &period) < 0) { | ||||
|             info("error: can't achieve requested scan rate (%lf Hz)\n", | ||||
|                 desired_rate); | ||||
|             return 1; | ||||
|         } | ||||
|     } else { | ||||
|         if (ue9_choose_scan(desired_rate, &actual_rate,  | ||||
|                     &scanconfig, &scaninterval) < 0) { | ||||
|             info("error: can't achieve requested scan rate (%lf Hz)\n", | ||||
|                 desired_rate); | ||||
|             return 1; | ||||
|         } | ||||
|     } | ||||
|      | ||||
| 	if (donerdjack) { | ||||
| 		if (nerdjack_choose_scan(desired_rate, &actual_rate, &period) < | ||||
| 		    0) { | ||||
| 			info("error: can't achieve requested scan rate (%lf Hz)\n", desired_rate); | ||||
| 		} | ||||
| 	} else { | ||||
| 		if (ue9_choose_scan(desired_rate, &actual_rate, | ||||
| 				    &scanconfig, &scaninterval) < 0) { | ||||
| 			info("error: can't achieve requested scan rate (%lf Hz)\n", desired_rate); | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	if ((desired_rate != actual_rate) || verb_count) | ||||
| 	if ((desired_rate != actual_rate) || verb_count) { | ||||
| 		info("Actual scanrate is %lf Hz\n", actual_rate); | ||||
| 		info("Period is %ld\n", period); | ||||
| 	} | ||||
|  | ||||
| 	if (verb_count && lines) { | ||||
| 		info("Stopping capture after %d lines\n", lines); | ||||
| @@ -272,35 +422,63 @@ int main(int argc, char *argv[]) | ||||
|  | ||||
| 	signal(SIGINT, handle_sig); | ||||
| 	signal(SIGTERM, handle_sig); | ||||
|      | ||||
|     if (detect) { | ||||
|         info("Autodetecting NerdJack address\n"); | ||||
|         free(address); | ||||
|         if(nerdjack_detect(address) < 0) { | ||||
|             info("Error with autodetection\n"); | ||||
|         } else { | ||||
|             info("Found NerdJack at address: %s\n",address); | ||||
|         } | ||||
|     } | ||||
|          | ||||
|  | ||||
| 	if (detect) { | ||||
| 		info("Autodetecting NerdJack address\n"); | ||||
| 		free(address); | ||||
| 		if (nerdjack_detect(address) < 0) { | ||||
| 			info("Error with autodetection\n"); | ||||
| 			goto printhelp; | ||||
| 		} else { | ||||
| 			info("Found NerdJack at address: %s\n", address); | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	for (;;) { | ||||
| 		int ret; | ||||
|         if(nerdjack) { | ||||
|             ret = nerdDoStream(address, channel_list, channel_count, precision, period, convert, lines); | ||||
|             verb("nerdDoStream returned %d\n", ret); | ||||
|          | ||||
|         } else { | ||||
|             ret = doStream(address, scanconfig, scaninterval, | ||||
|                     channel_list, channel_count, convert, | ||||
|                     lines); | ||||
|             verb("doStream returned %d\n", ret); | ||||
|         } | ||||
| 		if (donerdjack) { | ||||
| 			ret = | ||||
| 			    nerdDoStream(address, channel_list, channel_count, | ||||
| 					 precision, period, convert, lines, | ||||
| 					 showmem); | ||||
| 			verb("nerdDoStream returned %d\n", ret); | ||||
|  | ||||
| 		} else { | ||||
| 			ret = doStream(address, scanconfig, scaninterval, | ||||
| 				       channel_list, channel_count, | ||||
| 				       timer_mode_list, timer_mode_count, timer_divisor, | ||||
| 				       convert, lines); | ||||
| 			verb("doStream returned %d\n", ret); | ||||
| 		} | ||||
| 		if (oneshot) | ||||
| 			break; | ||||
|  | ||||
| 		if (ret == 0) | ||||
| 		        break; | ||||
| 			break; | ||||
|  | ||||
| 		//Neither options specified at command line and first time through. | ||||
| 		//Try LabJack | ||||
| 		if (ret == -ENOTCONN && donerdjack && !labjack && !nerdjack) { | ||||
| 			info("Could not connect NerdJack...Trying LabJack\n"); | ||||
| 			donerdjack = 0; | ||||
| 			goto doneparse; | ||||
| 		} | ||||
| 		//Neither option supplied, no address, and second time through. | ||||
| 		//Try autodetection | ||||
| 		if (ret == -ENOTCONN && !donerdjack && !labjack && !nerdjack | ||||
| 		    && !addressSpecified) { | ||||
| 			info("Could not connect LabJack...Trying to autodetect Nerdjack\n"); | ||||
| 			detect = 1; | ||||
| 			donerdjack = 1; | ||||
| 			goto doneparse; | ||||
| 		} | ||||
|  | ||||
| 		if (ret == -ENOTCONN && nerdjack && !detect | ||||
| 		    && !addressSpecified) { | ||||
| 			info("Could not reach NerdJack...Trying to autodetect\n"); | ||||
| 			detect = 1; | ||||
| 			goto doneparse; | ||||
| 		} | ||||
|  | ||||
| 		if (ret == -ENOTCONN && !forceretry) { | ||||
| 			info("Initial connection failed, giving up\n"); | ||||
| @@ -321,34 +499,87 @@ int main(int argc, char *argv[]) | ||||
| 	return 0; | ||||
| } | ||||
|  | ||||
| int nerdDoStream(const char *address, int *channel_list, int channel_count, int precision,  | ||||
|             unsigned short period, int convert, int lines) | ||||
| int | ||||
| nerdDoStream(const char *address, int *channel_list, int channel_count, | ||||
| 	     int precision, unsigned long period, int convert, int lines, | ||||
| 	     int showmem) | ||||
| { | ||||
| 	int retval = -EAGAIN; | ||||
| 	int fd_data; | ||||
| 	static int first_call = 1; | ||||
|     char command[13]; | ||||
| 	static int started = 0; | ||||
| 	static int wasreset = 0; | ||||
| 	getPacket command; | ||||
| 	static unsigned short currentcount = 0; | ||||
|  tryagain: | ||||
|  | ||||
| 	/* Open connection.  If this fails, and this is the | ||||
|   	   first attempt, return a different error code so we give up. */ | ||||
| 	//If this is the first time, set up acquisition | ||||
| 	//Otherwise try to resume the previous one | ||||
| 	if (started == 0) { | ||||
| 		if (nerd_generate_command | ||||
| 		    (&command, channel_list, channel_count, precision, | ||||
| 		     period) < 0) { | ||||
| 			info("Failed to create configuration command\n"); | ||||
| 			goto out; | ||||
| 		} | ||||
|  | ||||
| 		if (nerd_send_command(address, "STOP", 4) < 0) { | ||||
| 			if (first_call) { | ||||
| 				retval = -ENOTCONN; | ||||
| 				if (verb_count) | ||||
| 					info("Failed to send STOP command\n"); | ||||
| 			} else { | ||||
| 				info("Failed to send STOP command\n"); | ||||
| 			} | ||||
| 			goto out; | ||||
| 		} | ||||
|  | ||||
| 		if (nerd_send_command(address, &command, sizeof(command)) < 0) { | ||||
| 			info("Failed to send GET command\n"); | ||||
| 			goto out; | ||||
| 		} | ||||
|  | ||||
| 	} else { | ||||
| 		//If we had a transmission in progress, send a command to resume from there | ||||
| 		char cmdbuf[10]; | ||||
| 		sprintf(cmdbuf, "SETC%05hd", currentcount); | ||||
| 		retval = nerd_send_command(address, cmdbuf, strlen(cmdbuf)); | ||||
| 		if (retval == -4) { | ||||
| 			info("NerdJack was reset\n"); | ||||
| 			//Assume we have not started yet, reset on this side. | ||||
| 			//If this routine is retried, start over | ||||
| 			printf("# NerdJack was reset here\n"); | ||||
| 			currentcount = 0; | ||||
| 			started = 0; | ||||
| 			wasreset = 1; | ||||
| 			goto tryagain; | ||||
| 		} else if (retval < 0) { | ||||
| 			info("Failed to send SETC command\n"); | ||||
| 			goto out; | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	//The transmission has begun | ||||
| 	started = 1; | ||||
|  | ||||
| 	/* Open connection */ | ||||
| 	fd_data = nerd_open(address, NERDJACK_DATA_PORT); | ||||
| 	if (fd_data < 0) { | ||||
| 		info("Connect failed: %s:%d\n", address, NERDJACK_DATA_PORT); | ||||
| 		if (first_call) | ||||
| 			retval = -ENOTCONN; | ||||
| 		goto out; | ||||
| 	} | ||||
| 	first_call = 0; | ||||
|      | ||||
|     if (nerd_generate_command(command, channel_list, channel_count, precision, period) < 0) { | ||||
|         info("Failed to create configuration command\n"); | ||||
|         goto out1; | ||||
|     } | ||||
|      | ||||
|     if (nerd_data_stream(fd_data, command, channel_count, channel_list, precision, convert, lines) < 0) { | ||||
|         info("Failed to open data stream\n"); | ||||
|         goto out1; | ||||
|     } | ||||
|  | ||||
| 	retval = nerd_data_stream | ||||
| 	    (fd_data, channel_count, channel_list, precision, convert, lines, | ||||
| 	     showmem, ¤tcount, period, wasreset); | ||||
| 	wasreset = 0; | ||||
| 	if (retval == -3) { | ||||
| 		retval = 0; | ||||
| 	} | ||||
| 	if (retval < 0) { | ||||
| 		info("Failed to open data stream\n"); | ||||
| 		goto out1; | ||||
| 	} | ||||
|  | ||||
| 	info("Stream finished\n"); | ||||
| 	retval = 0; | ||||
| @@ -356,15 +587,20 @@ int nerdDoStream(const char *address, int *channel_list, int channel_count, int | ||||
|  out1: | ||||
| 	nerd_close_conn(fd_data); | ||||
|  out: | ||||
| 	//We've tried communicating, so this is not the first call anymore | ||||
| 	first_call = 0; | ||||
| 	return retval; | ||||
| } | ||||
|  | ||||
| int doStream(const char *address, uint8_t scanconfig, uint16_t scaninterval, | ||||
| 	     int *channel_list, int channel_count, int convert, int lines) | ||||
| int | ||||
| doStream(const char *address, uint8_t scanconfig, uint16_t scaninterval, | ||||
| 	 int *channel_list, int channel_count,  | ||||
| 	 int *timer_mode_list, int timer_mode_count, int timer_divisor, | ||||
| 	 int convert, int lines) | ||||
| { | ||||
| 	int retval = -EAGAIN; | ||||
| 	int fd_cmd, fd_data; | ||||
|     int ret; | ||||
| 	int ret; | ||||
| 	static int first_call = 1; | ||||
| 	struct callbackInfo ci = { | ||||
| 		.convert = convert, | ||||
| @@ -372,7 +608,7 @@ int doStream(const char *address, uint8_t scanconfig, uint16_t scaninterval, | ||||
| 	}; | ||||
|  | ||||
| 	/* Open command connection.  If this fails, and this is the | ||||
|   	   first attempt, return a different error code so we give up. */ | ||||
| 	   first attempt, return a different error code so we give up. */ | ||||
| 	fd_cmd = ue9_open(address, UE9_COMMAND_PORT); | ||||
| 	if (fd_cmd < 0) { | ||||
| 		info("Connect failed: %s:%d\n", address, UE9_COMMAND_PORT); | ||||
| @@ -393,13 +629,21 @@ int doStream(const char *address, uint8_t scanconfig, uint16_t scaninterval, | ||||
| 		info("Connect failed: %s:%d\n", address, UE9_DATA_PORT); | ||||
| 		goto out1; | ||||
| 	} | ||||
| 	 | ||||
|  | ||||
| 	/* Get calibration */ | ||||
| 	if (ue9_get_calibration(fd_cmd, &ci.calib) < 0) { | ||||
| 		info("Failed to get device calibration\n"); | ||||
| 		goto out2; | ||||
| 	} | ||||
|  | ||||
| 	/* Set timer configuration */ | ||||
| 	if (timer_mode_count && | ||||
| 	    ue9_timer_config(fd_cmd, timer_mode_list, timer_mode_count, | ||||
| 			     timer_divisor) < 0) { | ||||
| 		info("Failed to set timer configuration\n"); | ||||
| 		goto out2; | ||||
| 	}		 | ||||
|  | ||||
| 	/* Set stream configuration */ | ||||
| 	if (ue9_streamconfig_simple(fd_cmd, channel_list, channel_count, | ||||
| 				    scanconfig, scaninterval, | ||||
| @@ -415,7 +659,8 @@ int doStream(const char *address, uint8_t scanconfig, uint16_t scaninterval, | ||||
| 	} | ||||
|  | ||||
| 	/* Stream data */ | ||||
| 	ret = ue9_stream_data(fd_data, channel_count, data_callback, (void *)&ci); | ||||
| 	ret = | ||||
| 	    ue9_stream_data(fd_data, channel_count, data_callback, (void *)&ci); | ||||
| 	if (ret < 0) { | ||||
| 		info("Data stream failed with error %d\n", ret); | ||||
| 		goto out3; | ||||
| @@ -436,7 +681,7 @@ int doStream(const char *address, uint8_t scanconfig, uint16_t scaninterval, | ||||
| 	return retval; | ||||
| } | ||||
|  | ||||
| int data_callback(int channels, uint16_t *data, void *context) | ||||
| int data_callback(int channels, uint16_t * data, void *context) | ||||
| { | ||||
| 	int i; | ||||
| 	struct callbackInfo *ci = (struct callbackInfo *)context; | ||||
| @@ -444,22 +689,40 @@ int data_callback(int channels, uint16_t *data, void *context) | ||||
|  | ||||
| 	columns_left = channels; | ||||
| 	for (i = 0; i < channels; i++) { | ||||
| 		if (ci->convert) | ||||
| 			printf("%lf", ue9_binary_to_analog( | ||||
| 				       &ci->calib, UE9_BIPOLAR_GAIN1, 12,  | ||||
| 				       data[i])); | ||||
| 		else | ||||
| 			printf("%d", data[i]); | ||||
| 		switch (ci->convert) { | ||||
| 		case CONVERT_VOLTS: | ||||
| 			if (printf | ||||
| 			    ("%lf", | ||||
| 			     ue9_binary_to_analog(&ci->calib, UE9_BIPOLAR_GAIN1, | ||||
| 						  12, data[i])) < 0) | ||||
| 				goto bad; | ||||
| 			break; | ||||
| 		case CONVERT_HEX: | ||||
| 			if (printf("%04X", data[i]) < 0) | ||||
| 				goto bad; | ||||
| 			break; | ||||
| 		default: | ||||
| 		case CONVERT_DEC: | ||||
| 			if (printf("%d", data[i]) < 0) | ||||
| 				goto bad; | ||||
| 			break; | ||||
| 		} | ||||
| 		columns_left--; | ||||
| 		if (i < (channels - 1)) { | ||||
| 			putchar(' '); | ||||
| 			if (ci->convert != CONVERT_HEX && putchar(' ') < 0) | ||||
| 				goto bad; | ||||
| 		} else { | ||||
| 			putchar('\n'); | ||||
| 			if (putchar('\n') < 0) | ||||
| 				goto bad; | ||||
| 			lines++; | ||||
| 			if (ci->maxlines && lines >= ci->maxlines) | ||||
| 				return -1; | ||||
| 		} | ||||
| 	} | ||||
| 	 | ||||
|  | ||||
| 	return 0; | ||||
|  | ||||
|  bad: | ||||
| 	info("Output error (disk full?)\n"); | ||||
| 	return -3; | ||||
| } | ||||
|   | ||||
							
								
								
									
										8
									
								
								ethstream.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										8
									
								
								ethstream.h
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,8 @@ | ||||
| #ifndef ETHSTREAM_H | ||||
| #define ETHSTREAM_H | ||||
|  | ||||
| #define CONVERT_DEC 0 | ||||
| #define CONVERT_VOLTS 1 | ||||
| #define CONVERT_HEX 2 | ||||
|  | ||||
| #endif | ||||
							
								
								
									
										87
									
								
								ethstream.txt
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										87
									
								
								ethstream.txt
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,87 @@ | ||||
| ETHERSTREAM(1)                   User Commands                  ETHERSTREAM(1) | ||||
|  | ||||
|  | ||||
|  | ||||
| NAME | ||||
|        etherstream - manual page for etherstream 1.2 (2010-02-08) | ||||
|  | ||||
| SYNOPSIS | ||||
|        ethstream [options] | ||||
|                   ------- | ||||
|  | ||||
| DESCRIPTION | ||||
|        -a, --address string | ||||
|               host/address of device (192.168.1.209) | ||||
|  | ||||
|        -n, --numchannels n | ||||
|               sample the first N ADC channels (2) | ||||
|  | ||||
|        -C, --channels a,b,c | ||||
|               sample channels a, b, and c | ||||
|  | ||||
|        -r, --rate hz | ||||
|               sample each channel at this rate (8000.0) | ||||
|  | ||||
|        -L, --labjack | ||||
|               Force LabJack device | ||||
|  | ||||
|        -t, --timers a,b,c | ||||
|               set LabJack timer modes to a, b, and c | ||||
|  | ||||
|        -T, --timerdivisor n | ||||
|               set LabJack timer divisor to n | ||||
|  | ||||
|        -N, --nerdjack | ||||
|               Force NerdJack device | ||||
|  | ||||
|        -d, --detect | ||||
|               Detect NerdJack IP address | ||||
|  | ||||
|        -R, --range a,b | ||||
|               Set  range  on  NerdJack for channels 0-5,6-11 to either 5 or 10 | ||||
|               (10,10) | ||||
|  | ||||
|        -o, --oneshot | ||||
|               don’t retry in case of errors | ||||
|  | ||||
|        -f, --forceretry | ||||
|               retry no matter what happens | ||||
|  | ||||
|        -c, --convert | ||||
|               convert output to volts | ||||
|  | ||||
|        -H, --converthex | ||||
|               convert output to hex | ||||
|  | ||||
|        -m, --showmem | ||||
|               output memory stats with data (NJ only) | ||||
|  | ||||
|        -l, --lines num | ||||
|               if set, output this many lines and quit | ||||
|  | ||||
|        -h, --help | ||||
|               this help | ||||
|  | ||||
|        -v, --verbose | ||||
|               be verbose | ||||
|  | ||||
|        -V, --version | ||||
|               show version number and exit | ||||
|  | ||||
|        -i, --info | ||||
|               get info from device (NJ only) | ||||
|  | ||||
|        -X, --examples | ||||
|               show ethstream examples and exit | ||||
|  | ||||
|        Read data from the specified Labjack UE9 via Ethernet.  See README  for | ||||
|        details. | ||||
|  | ||||
| AUTHOR | ||||
|        Written    by   Jim   Paris   <jim@jtan.com>   and   Zachary   Clifford | ||||
|        <zacharyc@mit.edu>.  This program comes with no warranty  and  is  pro‐ | ||||
|        vided under the GPLv2. | ||||
|  | ||||
|  | ||||
|  | ||||
| etherstream 1.2 (2010-02-08)     February 2010                  ETHERSTREAM(1) | ||||
							
								
								
									
										74
									
								
								example.inc
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										74
									
								
								example.inc
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,74 @@ | ||||
|  | ||||
| char examplestring[] = "\n\ | ||||
| \n\ | ||||
|     Welcome to the NILM Ethstream examples.\n\ | ||||
| \n\ | ||||
| For the most part, typing \"ethstream\" by itself will sample the first\n\ | ||||
| two channels at 8 kHz on 10V range.  Press CTRL-C to terminate sampling.\n\ | ||||
| \n\ | ||||
| If you want current measurements on the first two phases of NILM\n\ | ||||
| with default sample rate of 8 kHz and 10V range:\n\ | ||||
| \n\ | ||||
|    ethstream -C 0,3\n\ | ||||
| \n\ | ||||
| The device is configured so that channels 0 through 2 are currents for\n\ | ||||
| the three phases and channels 3-5 are for voltages of the three phases.\n\ | ||||
| The current channels sample voltages that will depend on the DIP switch\n\ | ||||
| settings in the NILM box.  The DIP switch positions allow you to convert\n\ | ||||
| ethstream's readings to true current readings.\n\ | ||||
| \n\ | ||||
| If you want only currents at 16 kHz and 10V range:\n\ | ||||
| \n\ | ||||
|     ethstream -n 3 -r 16000\n\ | ||||
| \n\ | ||||
| The -n option samples a number of channels starting at 0. The rate can be\n\ | ||||
| at least 16000 if 12 channels are sampled , but it can do more if\n\ | ||||
| fewer channels are sampled.  The limiting factor is the highest channel\n\ | ||||
| sampled.  Sampling just the top channel (11) is as bad as sampling\n\ | ||||
| all 12 at once.\n\ | ||||
| Ethstream will warn if you approach the limits of the NerdJack with the\n\ | ||||
| given sampled channels.  Sampling outside the range of the NerdJack might\n\ | ||||
| result in corrupt data or crashing of the device.  There will be no\n\ | ||||
| permanent damage to NILM or NerdJack, but be aware of the possibility of\n\ | ||||
| data corruption.\n\ | ||||
| \n\ | ||||
| If you need a higher accuracy but lower range measurement on the voltages:\n\ | ||||
| \n\ | ||||
|     ethstream -R 5,10 -C 3,4,5\n\ | ||||
| \n\ | ||||
| The two numbers to the R command set the range to either 5V or 10V. Above,\n\ | ||||
| we are setting channels 0-5 to 5 V range and channels 6-11 to 10 V range.\n\ | ||||
| Channels 6-11 are unconnected, but they can have range set independently.\n\ | ||||
| \n\ | ||||
| All of the above examples output a digital number from 0 to 65535 with\n\ | ||||
| 65535 representing the highest range (5V or 10V).  0 represents the most\n\ | ||||
| negative range (-5V or -10V).  If you want conversion\n\ | ||||
| to volts for all six voltages and currents:\n\ | ||||
| \n\ | ||||
|     ethstream -c -C 0,3,1,4,2,5\n\ | ||||
| \n\ | ||||
| The channels will be output in the order given in the C command.  This\n\ | ||||
| command will group the current and voltage data by phase.\n\ | ||||
| \n\ | ||||
| If you are supplying data from ethstream to another program, you might\n\ | ||||
| want to dump its output to a file and terminate after a certain number of\n\ | ||||
| samples:\n\ | ||||
| \n\ | ||||
|     ethstream -n 6 -r 8000 - l 16000 > outfile.dat\n\ | ||||
| \n\ | ||||
| This will take 16000 samples at 8 kHz (2 seconds of acquisition) of all\n\ | ||||
| channels and write the data to outfile.dat.  This can be directly read\n\ | ||||
| by a package like MATLAB.\n\ | ||||
| \n\ | ||||
| If there are multiple NerdJacks or you have changed the TCP/IP settings\n\ | ||||
| from default, you might have to specify which one you want to talk to:\n\ | ||||
| \n\ | ||||
|     ethstream -a 192.168.1.210\n\ | ||||
| \n\ | ||||
| This will sample two channels at 8 kHz from the NerdJack at 192.168.1.210.\n\ | ||||
| This is the default \"1\" setting on the NerdJack.  If no address is\n\ | ||||
| specified, ethstream connects first to 192.168.1.209.  It then tries\n\ | ||||
| to autodetect the NerdJack.  This should find the device if you are on\n\ | ||||
| the same network, but it will get confused if there are multiple NerdJacks\n\ | ||||
| on the network.\n\ | ||||
| "; | ||||
							
								
								
									
										98
									
								
								ljconfig.c
									
									
									
									
									
								
							
							
						
						
									
										98
									
								
								ljconfig.c
									
									
									
									
									
								
							| @@ -1,98 +0,0 @@ | ||||
| /* | ||||
|  * Labjack Tools | ||||
|  * Copyright (c) 2003-2007 Jim Paris <jim@jtan.com> | ||||
|  * | ||||
|  * This is free software; you can redistribute it and/or modify it and | ||||
|  * it is provided under the terms of version 2 of the GNU General Public | ||||
|  * License as published by the Free Software Foundation; see COPYING. | ||||
|  */ | ||||
|  | ||||
| /* ljconfig: display/change comm/control processor configuration */ | ||||
|  | ||||
| #include <stdint.h> | ||||
| #include <stdlib.h> | ||||
| #include <stdio.h> | ||||
| #include <string.h> | ||||
| #include <errno.h> | ||||
| #include <unistd.h> | ||||
| #include "debug.h" | ||||
| #include "ue9.h" | ||||
| #include "ue9error.h" | ||||
| #include "opt.h" | ||||
| #include "version.h" | ||||
|  | ||||
| #define DEFAULT_HOST "192.168.1.209" | ||||
| #define UE9_COMMAND_PORT 52360 | ||||
|  | ||||
| struct options opt[] = { | ||||
| 	{ 'a', "address", "string", "host/address of UE9 (192.168.1.209)" }, | ||||
| 	{ 'h', "help", NULL, "this help" }, | ||||
| 	{ 'v', "verbose", NULL, "be verbose" }, | ||||
| 	{ 'V', "version", NULL, "show version number and exit" }, | ||||
| 	{ 0, NULL, NULL, NULL } | ||||
| }; | ||||
|  | ||||
| int main(int argc, char *argv[]) | ||||
| { | ||||
| 	int optind; | ||||
| 	char *optarg; | ||||
| 	char c; | ||||
| 	FILE *help = stderr; | ||||
| 	char *address = strdup(DEFAULT_HOST); | ||||
| 	int fd; | ||||
| 	int ret; | ||||
|  | ||||
| 	/* Parse arguments */ | ||||
| 	opt_init(&optind); | ||||
| 	while ((c = opt_parse(argc, argv, &optind, &optarg, opt)) != 0) { | ||||
| 		switch (c) { | ||||
| 		case 'a': | ||||
| 			free(address); | ||||
| 			address = strdup(optarg); | ||||
| 			break; | ||||
| 		case 'v': | ||||
| 			verb_count++; | ||||
| 			break; | ||||
| 		case 'V': | ||||
| 			printf("ljconfig " VERSION "\n"); | ||||
| 			printf("Written by Jim Paris <jim@jtan.com>\n"); | ||||
| 			printf("This program comes with no warranty and is " | ||||
| 			       "provided under the GPLv2.\n"); | ||||
| 			return 0; | ||||
| 			break; | ||||
| 		case 'h': | ||||
| 			help = stdout; | ||||
| 		default: | ||||
| 		printhelp: | ||||
| 			fprintf(help, "Usage: %s [options]\n", *argv); | ||||
| 			opt_help(opt, help); | ||||
| 			fprintf(help, "Displays/changes Labjack UE9 config.\n"); | ||||
| 			return (help == stdout) ? 0 : 1; | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	if(optind<argc) { | ||||
| 		info("Error: too many arguments (%s)\n\n", argv[optind]); | ||||
| 		goto printhelp; | ||||
| 	} | ||||
| 	 | ||||
| 	ret = 1; | ||||
|  | ||||
| 	/* Open */ | ||||
| 	fd = ue9_open(address, UE9_COMMAND_PORT); | ||||
| 	if (fd < 0) { | ||||
| 		info("Connect failed: %s:%d\n", address, UE9_COMMAND_PORT); | ||||
| 		goto out0; | ||||
| 	} | ||||
|  | ||||
| 	goto out1; | ||||
| 	 | ||||
| 	 | ||||
| 	ret = 0; | ||||
|  out1: | ||||
| 	/* Close */ | ||||
| 	ue9_close(fd); | ||||
|  out0: | ||||
| 	return ret; | ||||
| } | ||||
|  | ||||
							
								
								
									
										67
									
								
								ljtest.c
									
									
									
									
									
								
							
							
						
						
									
										67
									
								
								ljtest.c
									
									
									
									
									
								
							| @@ -1,67 +0,0 @@ | ||||
| /* | ||||
|  * Labjack Tools | ||||
|  * Copyright (c) 2003-2007 Jim Paris <jim@jtan.com> | ||||
|  * | ||||
|  * This is free software; you can redistribute it and/or modify it and | ||||
|  * it is provided under the terms of version 2 of the GNU General Public | ||||
|  * License as published by the Free Software Foundation; see COPYING. | ||||
|  */ | ||||
|  | ||||
| #include <stdint.h> | ||||
| #include <stdlib.h> | ||||
| #include <stdio.h> | ||||
| #include <string.h> | ||||
| #include <errno.h> | ||||
| #include "debug.h" | ||||
| #include "ue9.h" | ||||
| #include "compat.h" | ||||
|  | ||||
| int main(int argc, char *argv[]) | ||||
| { | ||||
| 	int fd_cmd; | ||||
| 	struct ue9Calibration calib; | ||||
|  | ||||
| 	verb_count = 2; | ||||
|  | ||||
| 	fd_cmd = ue9_open("192.168.1.209", 52360); | ||||
| 	if (fd_cmd < 0) { | ||||
| 		fprintf(stderr, "ue9_open: %s\n", | ||||
| 			compat_strerror(errno)); | ||||
| 		return 1; | ||||
| 	} | ||||
|  | ||||
| 	if (ue9_get_calibration(fd_cmd, &calib) < 0) { | ||||
| 		fprintf(stderr, "ue9_get_calibration: %s\n",  | ||||
| 			compat_strerror(errno)); | ||||
| 		return 1; | ||||
| 	} | ||||
|  | ||||
| 	printf("double unipolarSlope[0] = %lf\n", calib.unipolarSlope[0]); | ||||
| 	printf("double unipolarSlope[1] = %lf\n", calib.unipolarSlope[1]); | ||||
| 	printf("double unipolarSlope[2] = %lf\n", calib.unipolarSlope[2]); | ||||
| 	printf("double unipolarSlope[3] = %lf\n", calib.unipolarSlope[3]); | ||||
| 	printf("double unipolarOffset[0] = %lf\n", calib.unipolarOffset[0]); | ||||
| 	printf("double unipolarOffset[1] = %lf\n", calib.unipolarOffset[1]); | ||||
| 	printf("double unipolarOffset[2] = %lf\n", calib.unipolarOffset[2]); | ||||
| 	printf("double unipolarOffset[3] = %lf\n", calib.unipolarOffset[3]); | ||||
| 	printf("double bipolarSlope = %lf\n", calib.bipolarSlope); | ||||
| 	printf("double bipolarOffset = %lf\n", calib.bipolarOffset); | ||||
| 	printf("double DACSlope[0] = %lf\n", calib.DACSlope[0]); | ||||
| 	printf("double DACSlope[1] = %lf\n", calib.DACSlope[1]); | ||||
| 	printf("double DACOffset[0] = %lf\n", calib.DACOffset[0]); | ||||
| 	printf("double DACOffset[1] = %lf\n", calib.DACOffset[1]); | ||||
| 	printf("double tempSlope = %lf\n", calib.tempSlope); | ||||
| 	printf("double tempSlopeLow = %lf\n", calib.tempSlopeLow); | ||||
| 	printf("double calTemp = %lf\n", calib.calTemp); | ||||
| 	printf("double Vref = %lf\n", calib.Vref); | ||||
| 	printf("double VrefDiv2 = %lf\n", calib.VrefDiv2); | ||||
| 	printf("double VsSlope = %lf\n", calib.VsSlope); | ||||
| 	printf("double hiResUnipolarSlope = %lf\n", calib.hiResUnipolarSlope); | ||||
| 	printf("double hiResUnipolarOffset = %lf\n", calib.hiResUnipolarOffset); | ||||
| 	printf("double hiResBipolarSlope = %lf\n", calib.hiResBipolarSlope); | ||||
| 	printf("double hiResBipolarOffset = %lf\n", calib.hiResBipolarOffset); | ||||
|  | ||||
| 	ue9_close(fd_cmd); | ||||
|  | ||||
| 	return 0; | ||||
| } | ||||
							
								
								
									
										871
									
								
								nerdjack.c
									
									
									
									
									
								
							
							
						
						
									
										871
									
								
								nerdjack.c
									
									
									
									
									
								
							| @@ -13,7 +13,7 @@ | ||||
| #include <unistd.h> | ||||
| #include <stdio.h> | ||||
| #include <string.h> | ||||
| #include <sys/types.h> | ||||
|  | ||||
| #include <math.h> | ||||
|  | ||||
| #include "netutil.h" | ||||
| @@ -23,336 +23,671 @@ | ||||
| #include "nerdjack.h" | ||||
| #include "util.h" | ||||
| #include "netutil.h" | ||||
| #include "ethstream.h" | ||||
|  | ||||
| #define NERDJACK_TIMEOUT 5   /* Timeout for connect/send/recv, in seconds */ | ||||
| #define NERDJACK_TIMEOUT 5	/* Timeout for connect/send/recv, in seconds */ | ||||
|  | ||||
| #define NERD_HEADER_SIZE 8 | ||||
| #define MAX_SOCKETS 32 | ||||
|  | ||||
| typedef struct __attribute__ ((__packed__)) { | ||||
| 	unsigned char headerone; | ||||
| 	unsigned char headertwo; | ||||
| 	unsigned short packetNumber; | ||||
| 	unsigned short adcused; | ||||
| 	unsigned short packetsready; | ||||
| 	signed short data[NERDJACK_NUM_SAMPLES]; | ||||
| } dataPacket; | ||||
|  | ||||
| struct discovered_socket { | ||||
| 	int sock; | ||||
| 	uint32_t local_ip; | ||||
| 	uint32_t subnet_mask; | ||||
| }; | ||||
|  | ||||
| struct discover_t { | ||||
| 	struct discovered_socket socks[MAX_SOCKETS]; | ||||
| 	unsigned int sock_count; | ||||
| }; | ||||
|  | ||||
| /* Choose the best ScanConfig and ScanInterval parameters for the | ||||
|    desired scanrate.  Returns -1 if no valid config found */ | ||||
| int nerdjack_choose_scan(double desired_rate, double *actual_rate, int *period) | ||||
| int | ||||
| nerdjack_choose_scan(double desired_rate, double *actual_rate, | ||||
| 		     unsigned long *period) | ||||
| { | ||||
|      | ||||
| 	*period = round((double) NERDJACK_CLOCK_RATE / desired_rate); | ||||
|     * actual_rate = (double) NERDJACK_CLOCK_RATE / (double) *period; | ||||
| 	if(*actual_rate != desired_rate) { | ||||
| 	//The ffffe is because of a silicon bug.  The last bit is unusable in all | ||||
| 	//devices so far.  It is worked around on the chip, but giving it exactly | ||||
| 	//0xfffff would cause the workaround code to roll over. | ||||
| 	*period = floor((double)NERDJACK_CLOCK_RATE / desired_rate); | ||||
| 	if (*period > 0x0ffffe) { | ||||
| 		info("Cannot sample that slowly\n"); | ||||
| 		*actual_rate = (double)NERDJACK_CLOCK_RATE / (double)0x0ffffe; | ||||
| 		*period = 0x0ffffe; | ||||
| 		return -1; | ||||
| 	} | ||||
|     return 0; | ||||
| 	//Period holds the period register for the NerdJack, so it needs to be right | ||||
| 	*actual_rate = (double)NERDJACK_CLOCK_RATE / (double)*period; | ||||
| 	if (*actual_rate != desired_rate) { | ||||
| 		return -1; | ||||
| 	} | ||||
| 	return 0; | ||||
| } | ||||
|  | ||||
| int nerdjack_detect(char * ipAddress) { | ||||
|    int32_t sock, receivesock; | ||||
|    struct sockaddr_in sa, receiveaddr, sFromAddr; | ||||
|    int bytes_sent, buffer_length; | ||||
|    char buffer[200]; | ||||
|    char incomingData[10]; | ||||
|    unsigned int lFromLen; | ||||
|  | ||||
|    sprintf(buffer, "TEST"); | ||||
|    buffer_length = strlen(buffer) + 1; | ||||
|     | ||||
|    net_init(); | ||||
|  | ||||
|    sock = socket(PF_INET, SOCK_DGRAM, IPPROTO_UDP); | ||||
|    receivesock = socket(PF_INET, SOCK_DGRAM, IPPROTO_UDP); | ||||
|     | ||||
|    	/* Set nonblocking */ | ||||
|    if (soblock(sock, 0) < 0) { | ||||
|       verb("can't set nonblocking\n"); | ||||
|       return -1; | ||||
|    } | ||||
|     | ||||
|     /* Set nonblocking */ | ||||
|    if (soblock(receivesock, 0) < 0) { | ||||
|       verb("can't set nonblocking\n"); | ||||
|       return -1; | ||||
|    } | ||||
|  | ||||
|  | ||||
|    int opt = 1; | ||||
|    setsockopt(sock,SOL_SOCKET,SO_BROADCAST,(void *) &opt,sizeof(int)); | ||||
|     | ||||
|    if((-1 == sock) || (-1 == receivesock)) /* if socket failed to initialize, exit */ | ||||
|    { | ||||
|       printf("Error Creating Socket\n"); | ||||
|       return -1; | ||||
|    } | ||||
|  | ||||
|    //Setup family for both sockets | ||||
|    sa.sin_family = PF_INET; | ||||
|    receiveaddr.sin_family = PF_INET; | ||||
|     | ||||
|    //Setup ports to send on DATA and receive on RECEIVE | ||||
|    receiveaddr.sin_port = htons(NERDJACK_UDP_RECEIVE_PORT); | ||||
|    sa.sin_port = htons(NERDJACK_DATA_PORT); | ||||
|     | ||||
|    //Receive from any IP address, Will send to broadcast | ||||
|    receiveaddr.sin_addr.s_addr = INADDR_ANY; | ||||
|    sa.sin_addr.s_addr = INADDR_BROADCAST; | ||||
|  | ||||
|    bind(receivesock,(struct sockaddr*) &receiveaddr, sizeof(struct sockaddr_in)); | ||||
|  | ||||
|    bytes_sent = sendto(sock, buffer, buffer_length, 0,(struct sockaddr*) &sa, sizeof(struct sockaddr_in) ); | ||||
|    if(bytes_sent < 0) { | ||||
|      printf("Error sending packet: %s\n", strerror(errno) ); | ||||
| 	 return -1; | ||||
| /** | ||||
| * Create a discovered socket and add it to the socket list structure. | ||||
| * All sockets in the structure should be created, bound, and ready for broadcasting | ||||
| */ | ||||
| static int discovered_sock_create(struct discover_t *ds, uint32_t local_ip, | ||||
| 				  uint32_t subnet_mask) | ||||
| { | ||||
| 	if (ds->sock_count >= MAX_SOCKETS) { | ||||
| 		return 0; | ||||
| 	} | ||||
|  | ||||
|    lFromLen = sizeof(sFromAddr); | ||||
|     | ||||
|    if(0 > recvfrom_timeout(receivesock, incomingData, sizeof(incomingData),0,(struct sockaddr *) &sFromAddr, &lFromLen, | ||||
|                     & (struct timeval) { .tv_sec = NERDJACK_TIMEOUT })) { | ||||
|      | ||||
|         return -1; | ||||
|     } | ||||
|     | ||||
|    ipAddress = malloc(INET_ADDRSTRLEN); | ||||
|     | ||||
|    //It isn't ipv6 friendly, but inet_ntop isn't on Windows... | ||||
|    strcpy(ipAddress, inet_ntoa(sFromAddr.sin_addr)); | ||||
| 	/* Create socket. */ | ||||
| 	int sock = (int)socket(AF_INET, SOCK_DGRAM, 0); | ||||
| 	if (sock == -1) { | ||||
| 		return 0; | ||||
| 	} | ||||
|  | ||||
|    close(sock); /* close the socket */ | ||||
|    close(receivesock); | ||||
|    return 0; | ||||
| 	/* Allow broadcast. */ | ||||
| 	int sock_opt = 1; | ||||
| 	setsockopt(sock, SOL_SOCKET, SO_BROADCAST, (char *)&sock_opt, | ||||
| 		   sizeof(sock_opt)); | ||||
|  | ||||
| 	/* Set nonblocking */ | ||||
| 	if (soblock(sock, 0) < 0) { | ||||
| 		verb("can't set nonblocking\n"); | ||||
| 		return 0; | ||||
| 	} | ||||
|  | ||||
| 	/* Bind socket. */ | ||||
| 	struct sockaddr_in sock_addr; | ||||
| 	memset(&sock_addr, 0, sizeof(sock_addr)); | ||||
| 	sock_addr.sin_family = AF_INET; | ||||
| 	sock_addr.sin_addr.s_addr = htonl(local_ip); | ||||
| 	sock_addr.sin_port = htons(0); | ||||
| 	if (bind(sock, (struct sockaddr *)&sock_addr, sizeof(sock_addr)) != 0) { | ||||
| 		close(sock); | ||||
| 		return 0; | ||||
| 	} | ||||
|  | ||||
| 	/* Write sock entry. */ | ||||
| 	struct discovered_socket *dss = &ds->socks[ds->sock_count++]; | ||||
| 	dss->sock = sock; | ||||
| 	dss->local_ip = local_ip; | ||||
| 	dss->subnet_mask = subnet_mask; | ||||
|  | ||||
| 	return 1; | ||||
| } | ||||
|  | ||||
| typedef struct { | ||||
| int numCopies; | ||||
| int * destlist; | ||||
| } deststruct; | ||||
|  | ||||
| int nerd_data_stream(int data_fd, char * command, int numChannels, int *channel_list, int precision, int convert, int lines) | ||||
| /** | ||||
|  * Enumerate all interfaces we can find and open sockets on each | ||||
|  */ | ||||
| #if defined(USE_IPHLPAPI) | ||||
| static void enumerate_interfaces(struct discover_t *ds) | ||||
| { | ||||
| 	unsigned char buf[NERDJACK_PACKET_SIZE]; | ||||
| 	PIP_ADAPTER_INFO pAdapterInfo = | ||||
| 	    (IP_ADAPTER_INFO *) malloc(sizeof(IP_ADAPTER_INFO)); | ||||
| 	ULONG ulOutBufLen = sizeof(IP_ADAPTER_INFO); | ||||
|  | ||||
| 	//int numGroups = NERDJACK_NUM_SAMPLES / numChannels; | ||||
| 	DWORD Ret = GetAdaptersInfo(pAdapterInfo, &ulOutBufLen); | ||||
| 	if (Ret != NO_ERROR) { | ||||
| 		free(pAdapterInfo); | ||||
| 		if (Ret != ERROR_BUFFER_OVERFLOW) { | ||||
| 			return; | ||||
| 		} | ||||
| 		pAdapterInfo = (IP_ADAPTER_INFO *) malloc(ulOutBufLen); | ||||
| 		Ret = GetAdaptersInfo(pAdapterInfo, &ulOutBufLen); | ||||
| 		if (Ret != NO_ERROR) { | ||||
| 			free(pAdapterInfo); | ||||
| 			return; | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
|     int index = 0; | ||||
|     //int totalread = 0; | ||||
|     int ret = 0; | ||||
|     int alignment = 0; | ||||
|     signed short datapoint = 0; | ||||
|     signed short dataline[NERDJACK_CHANNELS]; | ||||
|     long double voltline[NERDJACK_CHANNELS]; | ||||
|     deststruct destination[NERDJACK_CHANNELS]; | ||||
|     int tempdestlist[NERDJACK_CHANNELS]; | ||||
|     unsigned short currentcount = 0; | ||||
|     unsigned long memused = 0; | ||||
| 	unsigned short packetsready = 0; | ||||
| 	unsigned short adcused = 0; | ||||
| 	unsigned short tempshort = 0; | ||||
|     int charsread = 0; | ||||
|     int charsleft = 0; | ||||
|     int additionalread = 0; | ||||
|     int linesleft = lines; | ||||
| 	PIP_ADAPTER_INFO pAdapter = pAdapterInfo; | ||||
| 	while (pAdapter) { | ||||
| 		IP_ADDR_STRING *pIPAddr = &pAdapter->IpAddressList; | ||||
| 		while (pIPAddr) { | ||||
| 			uint32_t local_ip = | ||||
| 			    ntohl(inet_addr(pIPAddr->IpAddress.String)); | ||||
| 			uint32_t mask = | ||||
| 			    ntohl(inet_addr(pIPAddr->IpMask.String)); | ||||
|  | ||||
| 	int numgroups = 0; | ||||
| 	long double volts; | ||||
|      | ||||
|     int channels_left = numChannels; | ||||
|     int channelprocessing = 0; | ||||
|     int currentalign = 0; //Index into sampled channels | ||||
|     int i; | ||||
|     int numDuplicates = 0; | ||||
|      | ||||
|     //Loop through channel_list until all channels recognized | ||||
|     //start with channelprocessing = 0 and increment through channels. | ||||
|     //If a channel is found in the list set it up appropriately. | ||||
|     do { | ||||
|         //numduplicates = 0; | ||||
|         destination[currentalign].numCopies = 0; | ||||
|         for(i = 0; i < numChannels; i++) { | ||||
|             if(channelprocessing == channel_list[i]) { | ||||
|                 //destination[currentalign] = i; | ||||
|                 tempdestlist[destination[currentalign].numCopies] = i; | ||||
|                 if(destination[currentalign].numCopies > 0) { | ||||
|                     numDuplicates++; | ||||
|                 } | ||||
|                 destination[currentalign].numCopies++; | ||||
|                 //currentalign++; | ||||
|                 channels_left--; | ||||
|                 //break; | ||||
|             } | ||||
|         } | ||||
|          | ||||
|         if(destination[currentalign].numCopies > 0) { | ||||
|             destination[currentalign].destlist = malloc( destination[currentalign].numCopies * sizeof(int) ); | ||||
|             memcpy(destination[currentalign].destlist, tempdestlist, destination[currentalign].numCopies * sizeof(int)); | ||||
|             currentalign++; | ||||
|         } | ||||
|         channelprocessing++; | ||||
|     } while(channels_left > 0); | ||||
|      | ||||
|      | ||||
|     int numChannelsSampled = numChannels - numDuplicates; | ||||
|     int numGroups = NERDJACK_NUM_SAMPLES / numChannelsSampled; | ||||
| 			if (local_ip == 0) { | ||||
| 				pIPAddr = pIPAddr->Next; | ||||
| 				continue; | ||||
| 			} | ||||
|  | ||||
|     /* Send request */ | ||||
| 	ret = send_all_timeout(data_fd, command, strlen(command), 0,  | ||||
| 			       & (struct timeval) { .tv_sec = NERDJACK_TIMEOUT }); | ||||
| 	if (ret < 0 || ret != strlen(command)) { | ||||
| 			discovered_sock_create(ds, local_ip, mask); | ||||
| 			pIPAddr = pIPAddr->Next; | ||||
| 		} | ||||
|  | ||||
| 		pAdapter = pAdapter->Next; | ||||
| 	} | ||||
|  | ||||
| 	free(pAdapterInfo); | ||||
| } | ||||
|  | ||||
| #else | ||||
| static void enumerate_interfaces(struct discover_t *ds) | ||||
| { | ||||
| 	int fd = socket(AF_INET, SOCK_DGRAM, 0); | ||||
| 	if (fd == -1) { | ||||
| 		return; | ||||
| 	} | ||||
|  | ||||
| 	struct ifconf ifc; | ||||
| 	uint8_t buf[8192]; | ||||
| 	ifc.ifc_len = sizeof(buf); | ||||
| 	ifc.ifc_buf = (char *)buf; | ||||
|  | ||||
| 	memset(buf, 0, sizeof(buf)); | ||||
|  | ||||
| 	if (ioctl(fd, SIOCGIFCONF, &ifc) != 0) { | ||||
| 		close(fd); | ||||
| 		return; | ||||
| 	} | ||||
|  | ||||
| 	uint8_t *ptr = (uint8_t *) ifc.ifc_req; | ||||
| 	uint8_t *end = (uint8_t *) & ifc.ifc_buf[ifc.ifc_len]; | ||||
|  | ||||
| 	while (ptr <= end) { | ||||
| 		struct ifreq *ifr = (struct ifreq *)ptr; | ||||
| 		ptr += _SIZEOF_ADDR_IFREQ(*ifr); | ||||
|  | ||||
| 		if (ioctl(fd, SIOCGIFADDR, ifr) != 0) { | ||||
| 			continue; | ||||
| 		} | ||||
| 		struct sockaddr_in *addr_in = | ||||
| 		    (struct sockaddr_in *)&(ifr->ifr_addr); | ||||
| 		uint32_t local_ip = ntohl(addr_in->sin_addr.s_addr); | ||||
| 		if (local_ip == 0) { | ||||
| 			continue; | ||||
| 		} | ||||
|  | ||||
| 		if (ioctl(fd, SIOCGIFNETMASK, ifr) != 0) { | ||||
| 			continue; | ||||
| 		} | ||||
|  | ||||
| 		struct sockaddr_in *mask_in = | ||||
| 		    (struct sockaddr_in *)&(ifr->ifr_addr); | ||||
| 		uint32_t mask = ntohl(mask_in->sin_addr.s_addr); | ||||
|  | ||||
| 		discovered_sock_create(ds, local_ip, mask); | ||||
| 	} | ||||
| } | ||||
| #endif | ||||
| /** | ||||
|  * Close all sockets previously enumerated and free the struct | ||||
|  */ | ||||
| static void destroy_socks(struct discover_t *ds) | ||||
| { | ||||
| 	unsigned int i; | ||||
| 	for (i = 0; i < ds->sock_count; i++) { | ||||
| 		struct discovered_socket *dss = &ds->socks[i]; | ||||
| 		close(dss->sock); | ||||
| 	} | ||||
|  | ||||
| 	free(ds); | ||||
| } | ||||
|  | ||||
| /* Perform autodetection.  Returns 0 on success, -1 on error | ||||
|  * Sets ipAddress to the detected address | ||||
|  */ | ||||
| int nerdjack_detect(char *ipAddress) | ||||
| { | ||||
| 	int32_t receivesock; | ||||
| 	struct sockaddr_in sa, receiveaddr, sFromAddr; | ||||
| 	int buffer_length; | ||||
| 	char buffer[200]; | ||||
| 	char incomingData[10]; | ||||
| 	unsigned int lFromLen; | ||||
|  | ||||
| 	sprintf(buffer, "TEST"); | ||||
| 	buffer_length = strlen(buffer) + 1; | ||||
|  | ||||
| 	net_init(); | ||||
|  | ||||
| 	receivesock = socket(PF_INET, SOCK_DGRAM, IPPROTO_UDP); | ||||
|  | ||||
| 	/* Set nonblocking */ | ||||
| 	if (soblock(receivesock, 0) < 0) { | ||||
| 		verb("can't set nonblocking\n"); | ||||
| 		return -1; | ||||
| 	} | ||||
|  | ||||
| 	if (-1 == receivesock) {	/* if socket failed to initialize, exit */ | ||||
| 		verb("Error Creating Socket\n"); | ||||
| 		return -1; | ||||
| 	} | ||||
| 	//Setup family for both sockets | ||||
| 	sa.sin_family = PF_INET; | ||||
| 	receiveaddr.sin_family = PF_INET; | ||||
|  | ||||
| 	//Setup ports to send on DATA and receive on RECEIVE | ||||
| 	receiveaddr.sin_port = htons(NERDJACK_UDP_RECEIVE_PORT); | ||||
| 	sa.sin_port = htons(NERDJACK_DATA_PORT); | ||||
|  | ||||
| 	//Receive from any IP address | ||||
| 	receiveaddr.sin_addr.s_addr = INADDR_ANY; | ||||
|  | ||||
| 	bind(receivesock, (struct sockaddr *)&receiveaddr, | ||||
| 	     sizeof(struct sockaddr_in)); | ||||
|  | ||||
| 	struct discover_t *ds = | ||||
| 	    (struct discover_t *)calloc(1, sizeof(struct discover_t)); | ||||
| 	if (!ds) { | ||||
| 		return -1; | ||||
| 	} | ||||
|  | ||||
| 	/* Create a routable broadcast socket. */ | ||||
| 	if (!discovered_sock_create(ds, 0, 0)) { | ||||
| 		free(ds); | ||||
| 		return -1; | ||||
| 	} | ||||
|  | ||||
| 	/* Detect & create local sockets. */ | ||||
| 	enumerate_interfaces(ds); | ||||
|  | ||||
| 	/* | ||||
| 	 * Send subnet broadcast using each local ip socket. | ||||
| 	 * This will work with multiple separate 169.254.x.x interfaces. | ||||
| 	 */ | ||||
| 	unsigned int i; | ||||
| 	for (i = 0; i < ds->sock_count; i++) { | ||||
| 		struct discovered_socket *dss = &ds->socks[i]; | ||||
| 		uint32_t target_ip = dss->local_ip | ~dss->subnet_mask; | ||||
| 		sa.sin_addr.s_addr = htonl(target_ip); | ||||
| 		sendto(dss->sock, buffer, buffer_length, 0, | ||||
| 		       (struct sockaddr *)&sa, sizeof(struct sockaddr_in)); | ||||
| 	} | ||||
|  | ||||
| 	destroy_socks(ds); | ||||
|  | ||||
| 	lFromLen = sizeof(sFromAddr); | ||||
|  | ||||
| 	if (0 > | ||||
| 	    recvfrom_timeout(receivesock, incomingData, sizeof(incomingData), 0, | ||||
| 			     (struct sockaddr *)&sFromAddr, &lFromLen, | ||||
| 			     &(struct timeval) { | ||||
| 			     .tv_sec = NERDJACK_TIMEOUT})) { | ||||
| 		close(receivesock); | ||||
| 		return -1; | ||||
| 	} | ||||
|  | ||||
| 	ipAddress = malloc(INET_ADDRSTRLEN); | ||||
|  | ||||
| 	//It isn't ipv6 friendly, but inet_ntop isn't on Windows... | ||||
| 	strcpy(ipAddress, inet_ntoa(sFromAddr.sin_addr)); | ||||
|  | ||||
| 	close(receivesock); | ||||
| 	return 0; | ||||
| } | ||||
|  | ||||
|  /* | ||||
|   * Get the NerdJack version string and print it | ||||
|   */ | ||||
| int nerd_get_version(const char *address) | ||||
| { | ||||
| 	int ret, fd_command; | ||||
| 	char buf[200]; | ||||
| 	fd_command = nerd_open(address, NERDJACK_COMMAND_PORT); | ||||
| 	if (fd_command < 0) { | ||||
| 		info("Connect failed: %s:%d\n", address, NERDJACK_COMMAND_PORT); | ||||
| 		return -2; | ||||
| 	} | ||||
|  | ||||
| 	/* Send request */ | ||||
| 	ret = send_all_timeout(fd_command, "VERS", 4, 0, &(struct timeval) { | ||||
| 			       .tv_sec = NERDJACK_TIMEOUT}); | ||||
| 	if (ret < 0) { | ||||
| 		verb("short send %d\n", (int)ret); | ||||
| 		return -1; | ||||
| 	} | ||||
|  | ||||
| 	ret = recv_all_timeout(fd_command, buf, 200, 0, &(struct timeval) { | ||||
| 			       .tv_sec = NERDJACK_TIMEOUT}); | ||||
|  | ||||
| 	nerd_close_conn(fd_command); | ||||
|  | ||||
| 	if (ret < 0) { | ||||
| 		verb("Error receiving command\n"); | ||||
| 		return -1; | ||||
| 	} | ||||
| 	//Slice off the "OK" from the string | ||||
| 	buf[strlen(buf) - 2] = '\0'; | ||||
|  | ||||
| 	printf("%s\n", buf); | ||||
|  | ||||
| 	return 0; | ||||
| } | ||||
|  | ||||
| /* Send the given command to address.  The command should be something | ||||
|  * of the specified length.  This expects the NerdJack to reply with OK | ||||
|  * or NO | ||||
|  */ | ||||
| int nerd_send_command(const char *address, void *command, int length) | ||||
| { | ||||
| 	int ret, fd_command; | ||||
| 	char buf[3]; | ||||
| 	fd_command = nerd_open(address, NERDJACK_COMMAND_PORT); | ||||
| 	if (fd_command < 0) { | ||||
| 		info("Connect failed: %s:%d\n", address, NERDJACK_COMMAND_PORT); | ||||
| 		return -2; | ||||
| 	} | ||||
|  | ||||
| 	/* Send request */ | ||||
| 	ret = send_all_timeout(fd_command, command, length, 0, &(struct timeval) { | ||||
| 			       .tv_sec = NERDJACK_TIMEOUT}); | ||||
| 	if (ret < 0 || ret != length) { | ||||
| 		verb("short send %d\n", (int)ret); | ||||
| 		return -1; | ||||
| 	} | ||||
|  | ||||
| 	ret = recv_all_timeout(fd_command, buf, 3, 0, &(struct timeval) { | ||||
| 			       .tv_sec = NERDJACK_TIMEOUT}); | ||||
|  | ||||
| 	nerd_close_conn(fd_command); | ||||
|  | ||||
| 	if (ret < 0 || ret != 3) { | ||||
| 		verb("Error receiving OK for command\n"); | ||||
| 		return -1; | ||||
| 	} | ||||
|  | ||||
| 	if (0 != strcmp("OK", buf)) { | ||||
| 		verb("Did not receive OK.  Received %s\n", buf); | ||||
| 		return -4; | ||||
| 	} | ||||
|  | ||||
| 	return 0; | ||||
| } | ||||
|  | ||||
| int | ||||
| nerd_data_stream(int data_fd, int numChannels, int *channel_list, | ||||
| 		 int precision, int convert, int lines, int showmem, | ||||
| 		 unsigned short *currentcount, unsigned int period, | ||||
| 		 int wasreset) | ||||
| { | ||||
| 	//Variables that should persist across retries | ||||
| 	static dataPacket buf; | ||||
| 	static int linesleft = 0; | ||||
| 	static int linesdumped = 0; | ||||
|  | ||||
| 	//Variables essential to packet processing | ||||
| 	signed short datapoint = 0; | ||||
| 	int i; | ||||
|  | ||||
| 	int numChannelsSampled = channel_list[0] + 1; | ||||
|  | ||||
| 	//The number sampled will be the highest channel requested plus 1 | ||||
| 	//(i.e. channel 0 requested means 1 sampled) | ||||
| 	for (i = 0; i < numChannels; i++) { | ||||
| 		if (channel_list[i] + 1 > numChannelsSampled) | ||||
| 			numChannelsSampled = channel_list[i] + 1; | ||||
| 	} | ||||
|  | ||||
| 	double voltline[numChannels]; | ||||
|  | ||||
| 	unsigned short dataline[numChannels]; | ||||
|  | ||||
| 	unsigned short packetsready = 0; | ||||
| 	unsigned short adcused = 0; | ||||
| 	unsigned short tempshort = 0; | ||||
| 	int charsread = 0; | ||||
|  | ||||
| 	int numgroupsProcessed = 0; | ||||
| 	double volts; | ||||
|  | ||||
| 	//The timeout should be the expected time plus 60 seconds | ||||
| 	//This permits slower speeds to work properly | ||||
| 	unsigned int expectedtimeout = | ||||
| 	    (period * NERDJACK_NUM_SAMPLES / NERDJACK_CLOCK_RATE) + 60; | ||||
|  | ||||
| 	//Check to see if we're trying to resume | ||||
| 	//Don't blow away linesleft in that case | ||||
| 	if (lines != 0 && linesleft == 0) { | ||||
| 		linesleft = lines; | ||||
| 	} | ||||
| 	//If there was a reset, we still need to dump a line because of faulty PDCA start | ||||
| 	if (wasreset) { | ||||
| 		linesdumped = 0; | ||||
| 	} | ||||
| 	//If this is the first time called, warn the user if we're too fast | ||||
| 	if (linesdumped == 0) { | ||||
| 		if (period < (numChannelsSampled * 200 + 600)) { | ||||
| 			info("You are sampling close to the limit of NerdJack\n"); | ||||
| 			info("Sample fewer channels or sample slower\n"); | ||||
| 		} | ||||
| 	} | ||||
| 	//Now destination structure array is set as well as numDuplicates. | ||||
|  | ||||
| 	int totalGroups = NERDJACK_NUM_SAMPLES / numChannelsSampled; | ||||
|  | ||||
| 	//Loop forever to grab data | ||||
|     while((charsread = recv_all_timeout(data_fd,buf,NERDJACK_PACKET_SIZE,0, | ||||
|            & (struct timeval) { .tv_sec = NERDJACK_TIMEOUT }))){ | ||||
| 	while ((charsread = | ||||
| 		recv_all_timeout(data_fd, &buf, NERDJACK_PACKET_SIZE, 0, | ||||
| 				 &(struct timeval) { | ||||
| 				 .tv_sec = expectedtimeout}))) { | ||||
|  | ||||
| 		//We want a complete packet, so take the chars so far and keep waiting | ||||
| 		if(charsread != NERDJACK_PACKET_SIZE) { | ||||
| 			charsleft = NERDJACK_PACKET_SIZE - charsread; | ||||
| 			while(charsleft != 0){ | ||||
| 				additionalread = recv_all_timeout(data_fd,buf+charsread,charsleft,0, | ||||
|                      & (struct timeval) { .tv_sec = NERDJACK_TIMEOUT }); | ||||
| 				charsread = charsread + additionalread; | ||||
| 				charsleft = NERDJACK_PACKET_SIZE - charsread; | ||||
| 			} | ||||
| 		if (charsread != NERDJACK_PACKET_SIZE) { | ||||
| 			//There was a problem getting data.  Probably a closed | ||||
| 			//connection. | ||||
| 			info("Packet timed out or was too short\n"); | ||||
| 			return -2; | ||||
| 		} | ||||
|  | ||||
| 		//First check the header info | ||||
| 		if(buf[0] != 0xF0 || buf[1] != 0xAA) { | ||||
| 			printf("No Header info\n"); | ||||
| 		if (buf.headerone != 0xF0 || buf.headertwo != 0xAA) { | ||||
| 			info("No Header info\n"); | ||||
| 			return -1; | ||||
| 		} | ||||
|  | ||||
| 		//Check counter info to make sure not out of order | ||||
| 		tempshort = (buf[2] << 8) | buf[3]; | ||||
| 		if(tempshort != currentcount ){ | ||||
| 			printf("Count wrong. Expected %hd but got %hd\n", currentcount, tempshort); | ||||
| 		tempshort = ntohs(buf.packetNumber); | ||||
| 		if (tempshort != *currentcount) { | ||||
| 			info("Count wrong. Expected %hd but got %hd\n", | ||||
| 			     *currentcount, tempshort); | ||||
| 			return -1; | ||||
| 		} | ||||
|          | ||||
|         //Increment number of packets received | ||||
| 		currentcount++; | ||||
|          | ||||
|         //Process the rest of the header and update the index value to be pointing after it | ||||
| 		index = 12; | ||||
| 		memused = (buf[4] << 24) | (buf[5] << 16) | (buf[6] << 8) | (buf[7]); | ||||
| 		adcused = (buf[8] << 8) | (buf[9]); | ||||
| 		packetsready = (buf[10] << 8) | (buf[11]); | ||||
| 		alignment = 0; | ||||
| 		numgroups = 0; | ||||
|          | ||||
|         //While there is still more data in the packet, process it | ||||
| 		while(charsread > index) { | ||||
| 			datapoint = (buf[index] << 8 | buf[index+1]); | ||||
|             if(convert) { | ||||
| 				if(alignment <= 5) { | ||||
| 					volts = (long double) ( datapoint / 32767.0 ) * ((precision & 0x01) ? 5.0 : 10.0); | ||||
| 				} else { | ||||
| 					volts = (long double) (datapoint / 32767.0 ) * ((precision & 0x02) ? 5.0 : 10.0); | ||||
| 				} | ||||
|                 for(i = 0; i < destination[alignment].numCopies; i++) { | ||||
|                     voltline[destination[alignment].destlist[i]] = volts; | ||||
|                 } | ||||
| 			} else { | ||||
|                 for(i = 0; i < destination[alignment].numCopies; i++) { | ||||
|                     dataline[destination[alignment].destlist[i]] = datapoint; | ||||
|                 } | ||||
| 		//Increment number of packets received | ||||
| 		*currentcount = *currentcount + 1; | ||||
|  | ||||
| 		adcused = ntohs(buf.adcused); | ||||
| 		packetsready = ntohs(buf.packetsready); | ||||
| 		numgroupsProcessed = 0; | ||||
|  | ||||
| 		if (showmem) { | ||||
| 			printf("%hd %hd\n", adcused, packetsready); | ||||
| 			continue; | ||||
| 		} | ||||
| 		//While there is still more data in the packet, process it | ||||
| 		while (numgroupsProcessed < totalGroups) { | ||||
| 			//Poison the data structure | ||||
| 			switch (convert) { | ||||
| 			case CONVERT_VOLTS: | ||||
| 				memset(voltline, 0, | ||||
| 				       numChannels * sizeof(double)); | ||||
| 				break; | ||||
| 			default: | ||||
| 			case CONVERT_HEX: | ||||
| 			case CONVERT_DEC: | ||||
| 				memset(dataline, 0, | ||||
| 				       numChannels * sizeof(unsigned char)); | ||||
| 			} | ||||
|              | ||||
|             //Each point is two bytes, so increment index and total bytes read | ||||
| 			index++; | ||||
| 			index++; | ||||
|             alignment++; | ||||
| 			//totalread++; | ||||
|              | ||||
|              | ||||
| 			 | ||||
|              | ||||
|             //Since channel data is packed, we need to know when to insert a newline | ||||
| 			if(alignment == numChannelsSampled){ | ||||
|                 if(convert) { | ||||
|                     for(i = 0; i < numChannels; i++) { | ||||
|                         printf("%Lf ",voltline[i]); | ||||
|                     } | ||||
|                 } else { | ||||
|                     for(i = 0; i < numChannels; i++) { | ||||
|                         printf("%hd ",dataline[i]); | ||||
|                     } | ||||
|                 } | ||||
| 				printf("\n"); | ||||
| 				alignment = 0; | ||||
| 				numgroups++; | ||||
|                 if(lines != 0) { | ||||
|                     linesleft--; | ||||
|                     if(linesleft == 0) { | ||||
|                         return 0; | ||||
|                     } | ||||
|                 } | ||||
|                 //If numgroups so far is equal to the numGroups in a packet, this packet is done | ||||
| 				if(numgroups == numGroups) { | ||||
|  | ||||
| 			//Read in each group | ||||
| 			for (i = 0; i < numChannels; i++) { | ||||
| 				//Get the datapoint associated with the desired channel | ||||
| 				datapoint = | ||||
| 				    ntohs(buf.data[channel_list[i] + | ||||
| 						   numgroupsProcessed * | ||||
| 						   numChannelsSampled]); | ||||
|  | ||||
| 				//Place it into the line | ||||
| 				switch (convert) { | ||||
| 				case CONVERT_VOLTS: | ||||
| 					if (channel_list[i] <= 5) { | ||||
| 						volts = | ||||
| 						    (double)(datapoint / | ||||
| 							     32767.0) * | ||||
| 						    ((precision & 0x01) ? 5.0 : | ||||
| 						     10.0); | ||||
| 					} else { | ||||
| 						volts = | ||||
| 						    (double)(datapoint / | ||||
| 							     32767.0) * | ||||
| 						    ((precision & 0x02) ? 5.0 : | ||||
| 						     10.0); | ||||
| 					} | ||||
| 					voltline[i] = volts; | ||||
| 					break; | ||||
| 				default: | ||||
| 				case CONVERT_HEX: | ||||
| 				case CONVERT_DEC: | ||||
| 					dataline[i] = | ||||
| 					    (unsigned short)(datapoint - | ||||
| 							     INT16_MIN); | ||||
| 					break; | ||||
| 				} | ||||
| 			} | ||||
| 			//We want to dump the first line because it's usually spurious | ||||
| 			if (linesdumped != 0) { | ||||
| 				//Now print the group | ||||
| 				switch (convert) { | ||||
| 				case CONVERT_VOLTS: | ||||
| 					for (i = 0; i < numChannels; i++) { | ||||
| 						if (printf("%lf ", voltline[i]) | ||||
| 						    < 0) | ||||
| 							goto bad; | ||||
| 					} | ||||
| 					break; | ||||
| 				case CONVERT_HEX: | ||||
| 					for (i = 0; i < numChannels; i++) { | ||||
| 						if (printf("%04hX", dataline[i]) | ||||
| 						    < 0) | ||||
| 							goto bad; | ||||
| 					} | ||||
| 					break; | ||||
| 				default: | ||||
| 				case CONVERT_DEC: | ||||
| 					for (i = 0; i < numChannels; i++) { | ||||
| 						if (printf("%hu ", dataline[i]) | ||||
| 						    < 0) | ||||
| 							goto bad; | ||||
| 					} | ||||
| 					break; | ||||
| 				} | ||||
| 				if (printf("\n") < 0) | ||||
| 					goto bad; | ||||
|  | ||||
| 				//If we're counting lines, decrement them | ||||
| 				if (lines != 0) { | ||||
| 					linesleft--; | ||||
| 					if (linesleft == 0) { | ||||
| 						return 0; | ||||
| 					} | ||||
| 				} | ||||
|  | ||||
| 			} else { | ||||
| 				linesdumped = linesdumped + 1; | ||||
| 			} | ||||
|  | ||||
| 			//We've processed this group, so advance the counter | ||||
| 			numgroupsProcessed++; | ||||
|  | ||||
| 		} | ||||
| 		index = 0; | ||||
| 	} | ||||
|  | ||||
| 	return 0; | ||||
|  | ||||
|  bad: | ||||
| 	info("Output error (disk full?)\n"); | ||||
| 	return -3; | ||||
|  | ||||
| } | ||||
|  | ||||
| int nerd_open(const char *address,int port) { | ||||
|      | ||||
|     struct hostent *he; | ||||
|      | ||||
|     net_init(); | ||||
|      | ||||
|     int32_t i32SocketFD = socket(PF_INET, SOCK_STREAM, 0); | ||||
| /* Open a connection to the NerdJack */ | ||||
| int nerd_open(const char *address, int port) | ||||
| { | ||||
|  | ||||
|     if(-1 == i32SocketFD) | ||||
|     { | ||||
|       printf("cannot create socket"); | ||||
|       return -1; | ||||
|     } | ||||
|      | ||||
|     /* Set nonblocking */ | ||||
| 	struct hostent *he; | ||||
|  | ||||
| 	net_init(); | ||||
|  | ||||
| 	int32_t i32SocketFD = socket(PF_INET, SOCK_STREAM, 0); | ||||
|  | ||||
| 	if (-1 == i32SocketFD) { | ||||
| 		verb("cannot create socket"); | ||||
| 		return -1; | ||||
| 	} | ||||
|  | ||||
| 	/* Set nonblocking */ | ||||
| 	if (soblock(i32SocketFD, 0) < 0) { | ||||
| 		verb("can't set nonblocking\n"); | ||||
| 		return -1; | ||||
| 	} | ||||
|  | ||||
|     struct sockaddr_in stSockAddr; | ||||
|     memset(&stSockAddr, 0, sizeof(stSockAddr)); | ||||
| 	struct sockaddr_in stSockAddr; | ||||
| 	memset(&stSockAddr, 0, sizeof(stSockAddr)); | ||||
|  | ||||
|     stSockAddr.sin_family = AF_INET; | ||||
|     stSockAddr.sin_port = htons(port); | ||||
|      | ||||
|     he = gethostbyname(address); | ||||
| 	stSockAddr.sin_family = AF_INET; | ||||
| 	stSockAddr.sin_port = htons(port); | ||||
|  | ||||
| 	he = gethostbyname(address); | ||||
| 	if (he == NULL) { | ||||
| 		verb("gethostbyname(\"%s\") failed\n", address); | ||||
| 		return -1; | ||||
| 	} | ||||
| 	stSockAddr.sin_addr = *((struct in_addr *) he->h_addr); | ||||
| 	stSockAddr.sin_addr = *((struct in_addr *)he->h_addr); | ||||
|  | ||||
| 	debug("Resolved %s -> %s\n", address, inet_ntoa(stSockAddr.sin_addr)); | ||||
|      | ||||
|     /* Connect */ | ||||
| 	if (connect_timeout(i32SocketFD, (struct sockaddr *) &stSockAddr, sizeof(stSockAddr), | ||||
| 			    & (struct timeval) { .tv_sec = NERDJACK_TIMEOUT }) < 0) { | ||||
|  | ||||
| 	/* Connect */ | ||||
| 	if (connect_timeout | ||||
| 	    (i32SocketFD, (struct sockaddr *)&stSockAddr, sizeof(stSockAddr), | ||||
| 	     &(struct timeval) { | ||||
| 	     .tv_sec = 3}) < 0) { | ||||
| 		verb("connection to %s:%d failed: %s\n", | ||||
| 		     inet_ntoa(stSockAddr.sin_addr), port, compat_strerror(errno)); | ||||
| 		     inet_ntoa(stSockAddr.sin_addr), port, | ||||
| 		     compat_strerror(errno)); | ||||
| 		return -1; | ||||
| 	} | ||||
|      | ||||
|  | ||||
| 	return i32SocketFD; | ||||
| } | ||||
|  | ||||
| int nerd_generate_command(char * command, int * channel_list, int channel_count, int precision, | ||||
|     unsigned short period) { | ||||
|      | ||||
|     int channelbit = 0; | ||||
|     int i; | ||||
|      | ||||
|     for( i = 0; i < channel_count; i++) { | ||||
|         channelbit = channelbit | (0x1 << channel_list[i]); | ||||
|     } | ||||
| //Generate an appropriate sample initiation command | ||||
| int | ||||
| nerd_generate_command(getPacket * command, int *channel_list, | ||||
| 		      int channel_count, int precision, unsigned long period) | ||||
| { | ||||
|  | ||||
| 	short channelbit = 0; | ||||
| 	int i; | ||||
| 	int highestchannel = 0; | ||||
|  | ||||
| 	for (i = 0; i < channel_count; i++) { | ||||
| 		if (channel_list[i] > highestchannel) { | ||||
| 			highestchannel = channel_list[i]; | ||||
| 		} | ||||
| 		//channelbit = channelbit | (0x1 << channel_list[i]); | ||||
| 	} | ||||
|  | ||||
| 	for (i = 0; i <= highestchannel; i++) { | ||||
| 		channelbit = channelbit | (0x01 << i); | ||||
| 	} | ||||
|  | ||||
| 	command->word[0] = 'G'; | ||||
| 	command->word[1] = 'E'; | ||||
| 	command->word[2] = 'T'; | ||||
| 	command->word[3] = 'D'; | ||||
| 	command->channelbit = htons(channelbit); | ||||
| 	command->precision = precision; | ||||
| 	command->period = htonl(period); | ||||
| 	command->prescaler = 0; | ||||
|  | ||||
| 	return 0; | ||||
|  | ||||
|     sprintf(command,"GET%3.3X%d%5.5d", channelbit,precision,period); | ||||
|      | ||||
|     return 0; | ||||
|      | ||||
| } | ||||
|  | ||||
| int nerd_close_conn(int data_fd) | ||||
|   | ||||
							
								
								
									
										37
									
								
								nerdjack.h
									
									
									
									
									
								
							
							
						
						
									
										37
									
								
								nerdjack.h
									
									
									
									
									
								
							| @@ -16,29 +16,50 @@ | ||||
| #include "netutil.h" | ||||
|  | ||||
| #define NERDJACK_CHANNELS 12 | ||||
| #define NERDJACK_CLOCK_RATE 54000000 | ||||
| #define NERDJACK_CLOCK_RATE 66000000 | ||||
| #define NERDJACK_DATA_PORT 49155 | ||||
| #define NERDJACK_UDP_RECEIVE_PORT 49156 | ||||
| #define NERDJACK_COMMAND_PORT 49157 | ||||
|  | ||||
| #define NERDJACK_PACKET_SIZE 1460 | ||||
| #define NERDJACK_NUM_SAMPLES 724 | ||||
| #define NERDJACK_NUM_SAMPLES 726 | ||||
|  | ||||
| /* Packet structure used in message to start sampling on NerdJack */ | ||||
| typedef struct __attribute__ ((__packed__)) { | ||||
| 	char word[4]; | ||||
| 	unsigned long period; | ||||
| 	unsigned short channelbit; | ||||
| 	unsigned char precision; | ||||
| 	unsigned char prescaler; | ||||
| } getPacket; | ||||
|  | ||||
| /* Open/close TCP/IP connection to the NerdJack */ | ||||
| int nerd_open(const char *address,int port); | ||||
| int nerd_open(const char *address, int port); | ||||
| int nerd_close_conn(int data_fd); | ||||
|  | ||||
| /* Generate the command word for the NerdJack */ | ||||
| int nerd_generate_command(char * command, int * channel_list, int channel_count, int precision, | ||||
|     unsigned short period); | ||||
| int nerd_generate_command(getPacket * command, int *channel_list, | ||||
| 			  int channel_count, int precision, | ||||
| 			  unsigned long period); | ||||
|  | ||||
| /* Send given command to NerdJack */ | ||||
| int nerd_send_command(const char *address, void *command, int length); | ||||
|  | ||||
| /* Get the version string from NerdJack */ | ||||
| int nerd_get_version(const char *address); | ||||
|  | ||||
| /* Stream data out of the NerdJack */ | ||||
| int nerd_data_stream(int data_fd, char * command, int numChannels, int * channel_list, int precision, int convert, int lines); | ||||
| int nerd_data_stream(int data_fd, int numChannels, int *channel_list, | ||||
| 		     int precision, int convert, int lines, int showmem, | ||||
| 		     unsigned short *currentcount, unsigned int period, | ||||
| 		     int wasreset); | ||||
|  | ||||
| /* Detect the IP Address of the NerdJack and return in ipAddress */ | ||||
| int nerdjack_detect(char * ipAddress); | ||||
| int nerdjack_detect(char *ipAddress); | ||||
|  | ||||
| /* Choose the best ScanConfig and ScanInterval parameters for the | ||||
|    desired scanrate.  Returns -1 if no valid config found */ | ||||
| int nerdjack_choose_scan(double desired_rate, double *actual_rate, int *period); | ||||
| int nerdjack_choose_scan(double desired_rate, double *actual_rate, | ||||
| 			 unsigned long *period); | ||||
|  | ||||
| #endif | ||||
|   | ||||
							
								
								
									
										43
									
								
								netutil.c
									
									
									
									
									
								
							
							
						
						
									
										43
									
								
								netutil.c
									
									
									
									
									
								
							| @@ -39,15 +39,15 @@ int soblock(int socket, int blocking) | ||||
| 	/* Set flags */ | ||||
| 	if (fcntl(socket, F_SETFL, sockopt) != 0) | ||||
| 		return -1; | ||||
| 	 | ||||
|  | ||||
| 	return 0; | ||||
| #endif | ||||
| } | ||||
|  | ||||
|  | ||||
| /* Like connect(2), but with a timeout.  Socket must be non-blocking. */ | ||||
| int connect_timeout(int s, const struct sockaddr *serv_addr, socklen_t addrlen, | ||||
| 		    struct timeval *timeout) | ||||
| int | ||||
| connect_timeout(int s, const struct sockaddr *serv_addr, socklen_t addrlen, | ||||
| 		struct timeval *timeout) | ||||
| { | ||||
| 	int ret; | ||||
| 	fd_set writefds; | ||||
| @@ -60,7 +60,7 @@ int connect_timeout(int s, const struct sockaddr *serv_addr, socklen_t addrlen, | ||||
|  | ||||
| 	if (ret == 0) { | ||||
| 		/* Success */ | ||||
| 		return 0;    | ||||
| 		return 0; | ||||
| 	} | ||||
|  | ||||
| 	/* Check for immediate failure */ | ||||
| @@ -70,7 +70,7 @@ int connect_timeout(int s, const struct sockaddr *serv_addr, socklen_t addrlen, | ||||
| 		return -1; | ||||
| #else | ||||
| 	if (ret < 0 && errno != EINPROGRESS && errno != EALREADY) | ||||
| 		return -1;   | ||||
| 		return -1; | ||||
| #endif | ||||
|  | ||||
| 	/* In progress, wait for result. */ | ||||
| @@ -114,12 +114,13 @@ int connect_timeout(int s, const struct sockaddr *serv_addr, socklen_t addrlen, | ||||
| /* Like send(2), but with a timeout.  Socket must be non-blocking. | ||||
|    The timeout only applies if no data at all is sent -- this function | ||||
|    may still send less than requested. */ | ||||
| ssize_t send_timeout(int s, const void *buf, size_t len, int flags,  | ||||
| 		     struct timeval *timeout) | ||||
| ssize_t | ||||
| send_timeout(int s, const void *buf, size_t len, int flags, | ||||
| 	     struct timeval * timeout) | ||||
| { | ||||
| 	fd_set writefds; | ||||
| 	int ret; | ||||
| 	 | ||||
|  | ||||
| 	FD_ZERO(&writefds); | ||||
| 	FD_SET(s, &writefds); | ||||
| 	ret = select(s + 1, NULL, &writefds, NULL, timeout); | ||||
| @@ -139,12 +140,12 @@ ssize_t send_timeout(int s, const void *buf, size_t len, int flags, | ||||
| /* Like recv(2), but with a timeout.  Socket must be non-blocking.  | ||||
|    The timeout only applies if no data at all is received -- this | ||||
|    function may still return less than requested. */ | ||||
| ssize_t recv_timeout(int s, void *buf, size_t len, int flags,  | ||||
| 		     struct timeval *timeout) | ||||
| ssize_t | ||||
| recv_timeout(int s, void *buf, size_t len, int flags, struct timeval * timeout) | ||||
| { | ||||
| 	fd_set readfds; | ||||
| 	int ret; | ||||
| 	 | ||||
|  | ||||
| 	FD_ZERO(&readfds); | ||||
| 	FD_SET(s, &readfds); | ||||
| 	ret = select(s + 1, &readfds, NULL, NULL, timeout); | ||||
| @@ -164,12 +165,14 @@ ssize_t recv_timeout(int s, void *buf, size_t len, int flags, | ||||
| /* Like recvfrom(2), but with a timeout.  Socket must be non-blocking.  | ||||
|    The timeout only applies if no data at all is received -- this | ||||
|    function may still return less than requested. */ | ||||
| ssize_t recvfrom_timeout(int s, void *buf, size_t len, int flags, struct sockaddr *address, socklen_t *address_len, | ||||
| 		     struct timeval *timeout) | ||||
| ssize_t | ||||
| recvfrom_timeout(int s, void *buf, size_t len, int flags, | ||||
| 		 struct sockaddr * address, socklen_t * address_len, | ||||
| 		 struct timeval * timeout) | ||||
| { | ||||
| 	fd_set readfds; | ||||
| 	int ret; | ||||
| 	 | ||||
|  | ||||
| 	FD_ZERO(&readfds); | ||||
| 	FD_SET(s, &readfds); | ||||
| 	ret = select(s + 1, &readfds, NULL, NULL, timeout); | ||||
| @@ -189,8 +192,9 @@ ssize_t recvfrom_timeout(int s, void *buf, size_t len, int flags, struct sockadd | ||||
| /* Like send_timeout, but retries (with the same timeout) in case of | ||||
|    partial transfers.  This is a stronger attempt to send all | ||||
|    requested data. */ | ||||
| ssize_t send_all_timeout(int s, const void *buf, size_t len, int flags,  | ||||
| 			 struct timeval *timeout) | ||||
| ssize_t | ||||
| send_all_timeout(int s, const void *buf, size_t len, int flags, | ||||
| 		 struct timeval * timeout) | ||||
| { | ||||
| 	struct timeval tv; | ||||
| 	size_t left = len; | ||||
| @@ -217,8 +221,9 @@ ssize_t send_all_timeout(int s, const void *buf, size_t len, int flags, | ||||
| /* Like recv_timeout, but retries (with the same timeout) in case of | ||||
|    partial transfers.  This is a stronger attempt to recv all | ||||
|    requested data. */ | ||||
| ssize_t recv_all_timeout(int s, void *buf, size_t len, int flags,  | ||||
| 			 struct timeval *timeout) | ||||
| ssize_t | ||||
| recv_all_timeout(int s, void *buf, size_t len, int flags, | ||||
| 		 struct timeval * timeout) | ||||
| { | ||||
| 	struct timeval tv; | ||||
| 	size_t left = len; | ||||
|   | ||||
							
								
								
									
										27
									
								
								netutil.h
									
									
									
									
									
								
							
							
						
						
									
										27
									
								
								netutil.h
									
									
									
									
									
								
							| @@ -8,14 +8,22 @@ | ||||
| #ifdef __WIN32__ | ||||
| #  include <winsock2.h> | ||||
| #  include <ws2tcpip.h> | ||||
| #  define socklen_t int | ||||
| #  define socklen_t unsigned int | ||||
| #  define in_addr_t uint32_t | ||||
| #  define in_port_t uint16_t | ||||
| #  include <windows.h> | ||||
| #  include <iphlpapi.h> | ||||
| #  define USE_IPHLPAPI 1 | ||||
| #else | ||||
| #  include <sys/socket.h> | ||||
| #  include <netinet/in.h> | ||||
| #  include <arpa/inet.h> | ||||
| #  include <netdb.h> | ||||
| #  include <net/if.h> | ||||
| #  include <sys/ioctl.h> | ||||
| #ifndef _SIZEOF_ADDR_IFREQ | ||||
| #define _SIZEOF_ADDR_IFREQ(x) sizeof(x) | ||||
| #endif | ||||
| #endif | ||||
|  | ||||
| /* Initialize networking */ | ||||
| @@ -26,20 +34,21 @@ int soblock(int socket, int blocking); | ||||
|  | ||||
| /* Like send(2), recv(2), connect(2), but with timeouts.   | ||||
|    Socket must be O_NONBLOCK. */ | ||||
| int connect_timeout(int s, const struct sockaddr *serv_addr, socklen_t addrlen, | ||||
| 		    struct timeval *timeout); | ||||
| ssize_t send_timeout(int s, const void *buf, size_t len, int flags,  | ||||
| int connect_timeout(int s, const struct sockaddr *serv_addr, | ||||
| 		    socklen_t addrlen, struct timeval *timeout); | ||||
| ssize_t send_timeout(int s, const void *buf, size_t len, int flags, | ||||
| 		     struct timeval *timeout); | ||||
| ssize_t recv_timeout(int s, void *buf, size_t len, int flags,  | ||||
| 		     struct timeval *timeout); | ||||
| ssize_t recvfrom_timeout(int s, void *buf, size_t len, int flags, struct sockaddr *address, socklen_t *address_len, | ||||
| ssize_t recv_timeout(int s, void *buf, size_t len, int flags, | ||||
| 		     struct timeval *timeout); | ||||
| ssize_t recvfrom_timeout(int s, void *buf, size_t len, int flags, | ||||
| 			 struct sockaddr *address, socklen_t * address_len, | ||||
| 			 struct timeval *timeout); | ||||
|  | ||||
| /* Like send_timeout and recv_timeout, but they retry (with the same timeout) | ||||
|    in case of partial transfers, in order to try to transfer all data. */ | ||||
| ssize_t send_all_timeout(int s, const void *buf, size_t len, int flags,  | ||||
| ssize_t send_all_timeout(int s, const void *buf, size_t len, int flags, | ||||
| 			 struct timeval *timeout); | ||||
| ssize_t recv_all_timeout(int s, void *buf, size_t len, int flags,  | ||||
| ssize_t recv_all_timeout(int s, void *buf, size_t len, int flags, | ||||
| 			 struct timeval *timeout); | ||||
|  | ||||
| #endif | ||||
|   | ||||
							
								
								
									
										92
									
								
								opt.c
									
									
									
									
									
								
							
							
						
						
									
										92
									
								
								opt.c
									
									
									
									
									
								
							| @@ -11,70 +11,71 @@ | ||||
| #include <string.h> | ||||
| #include "opt.h" | ||||
|  | ||||
| void opt_init(int *optind) { | ||||
| 	*optind=0; | ||||
| void opt_init(int *optind) | ||||
| { | ||||
| 	*optind = 0; | ||||
| } | ||||
|  | ||||
| char opt_parse(int argc, char **argv, int *optind, char **optarg, | ||||
| 	       struct options *opt) { | ||||
| char | ||||
| opt_parse(int argc, char **argv, int *optind, char **optarg, | ||||
| 	  struct options *opt) | ||||
| { | ||||
| 	char c; | ||||
| 	int i; | ||||
| 	(*optind)++; | ||||
| 	if(*optind>=argc) | ||||
| 	if (*optind >= argc) | ||||
| 		return 0; | ||||
| 	 | ||||
| 	if(argv[*optind][0]=='-' &&  | ||||
| 	   argv[*optind][1]!='-' && | ||||
| 	   argv[*optind][1]!=0) { | ||||
|  | ||||
| 	if (argv[*optind][0] == '-' && | ||||
| 	    argv[*optind][1] != '-' && argv[*optind][1] != 0) { | ||||
| 		/* Short option (or a bunch of 'em) */ | ||||
| 		/* Save this and shift others over */ | ||||
| 		c=argv[*optind][1]; | ||||
| 		for(i=2;argv[*optind][i]!=0;i++) | ||||
| 			argv[*optind][i-1]=argv[*optind][i]; | ||||
| 		argv[*optind][i-1]=0; | ||||
| 		if(argv[*optind][1]!=0) | ||||
| 		c = argv[*optind][1]; | ||||
| 		for (i = 2; argv[*optind][i] != 0; i++) | ||||
| 			argv[*optind][i - 1] = argv[*optind][i]; | ||||
| 		argv[*optind][i - 1] = 0; | ||||
| 		if (argv[*optind][1] != 0) | ||||
| 			(*optind)--; | ||||
| 		/* Now find it */ | ||||
| 		for(i=0;opt[i].shortopt!=0;i++) | ||||
| 			if(opt[i].shortopt==c) | ||||
| 		for (i = 0; opt[i].shortopt != 0; i++) | ||||
| 			if (opt[i].shortopt == c) | ||||
| 				break; | ||||
| 		if(opt[i].shortopt==0) { | ||||
| 			fprintf(stderr,"Error: unknown option '-%c'\n",c); | ||||
| 		if (opt[i].shortopt == 0) { | ||||
| 			fprintf(stderr, "Error: unknown option '-%c'\n", c); | ||||
| 			return '?'; | ||||
| 		} | ||||
| 		if(opt[i].arg==NULL) | ||||
| 		if (opt[i].arg == NULL) | ||||
| 			return c; | ||||
| 		(*optind)++; | ||||
| 		if(*optind>=argc || (argv[*optind][0]=='-' && | ||||
| 			argv[*optind][1]!=0)) { | ||||
| 			fprintf(stderr,"Error: option '-%c' requires an " | ||||
| 				"argument\n",c); | ||||
| 		if (*optind >= argc || (argv[*optind][0] == '-' && | ||||
| 					argv[*optind][1] != 0)) { | ||||
| 			fprintf(stderr, "Error: option '-%c' requires an " | ||||
| 				"argument\n", c); | ||||
| 			return '?'; | ||||
| 		}  | ||||
| 		(*optarg)=argv[*optind]; | ||||
| 		} | ||||
| 		(*optarg) = argv[*optind]; | ||||
| 		return c; | ||||
| 	} else if(argv[*optind][0]=='-' && | ||||
| 		  argv[*optind][1]=='-' && | ||||
| 		  argv[*optind][2]!=0) { | ||||
| 	} else if (argv[*optind][0] == '-' && | ||||
| 		   argv[*optind][1] == '-' && argv[*optind][2] != 0) { | ||||
| 		/* Long option */ | ||||
| 		for(i=0;(c=opt[i].shortopt)!=0;i++) | ||||
| 			if(strcmp(opt[i].longopt,argv[*optind]+2)==0) | ||||
| 		for (i = 0; (c = opt[i].shortopt) != 0; i++) | ||||
| 			if (strcmp(opt[i].longopt, argv[*optind] + 2) == 0) | ||||
| 				break; | ||||
| 		if(opt[i].shortopt==0) { | ||||
| 			fprintf(stderr,"Error: unknown option '%s'\n", | ||||
| 		if (opt[i].shortopt == 0) { | ||||
| 			fprintf(stderr, "Error: unknown option '%s'\n", | ||||
| 				argv[*optind]); | ||||
| 			return '?'; | ||||
| 		} | ||||
| 		if(opt[i].arg==NULL) | ||||
| 		if (opt[i].arg == NULL) | ||||
| 			return c; | ||||
| 		(*optind)++; | ||||
| 		if(*optind>=argc || (argv[*optind][0]=='-' && | ||||
| 			argv[*optind][1]!=0)) { | ||||
| 			fprintf(stderr,"Error: option '%s' requires an " | ||||
| 				"argument\n",argv[*optind-1]); | ||||
| 		if (*optind >= argc || (argv[*optind][0] == '-' && | ||||
| 					argv[*optind][1] != 0)) { | ||||
| 			fprintf(stderr, "Error: option '%s' requires an " | ||||
| 				"argument\n", argv[*optind - 1]); | ||||
| 			return '?'; | ||||
| 		}  | ||||
| 		(*optarg)=argv[*optind]; | ||||
| 		} | ||||
| 		(*optarg) = argv[*optind]; | ||||
| 		return c; | ||||
| 	} else { | ||||
| 		/* End of options */ | ||||
| @@ -82,14 +83,15 @@ char opt_parse(int argc, char **argv, int *optind, char **optarg, | ||||
| 	} | ||||
| } | ||||
|  | ||||
| void opt_help(struct options *opt, FILE *out) { | ||||
| void opt_help(struct options *opt, FILE * out) | ||||
| { | ||||
| 	int i; | ||||
| 	int printed; | ||||
|  | ||||
| 	for(i=0;opt[i].shortopt!=0;i++) { | ||||
| 		fprintf(out,"  -%c, --%s%n",opt[i].shortopt, | ||||
| 			opt[i].longopt,&printed); | ||||
| 		fprintf(out," %-*s%s\n",30-printed, | ||||
| 			opt[i].arg?opt[i].arg:"",opt[i].help); | ||||
| 	for (i = 0; opt[i].shortopt != 0; i++) { | ||||
| 		fprintf(out, "  -%c, --%s%n", opt[i].shortopt, | ||||
| 			opt[i].longopt, &printed); | ||||
| 		fprintf(out, " %-*s%s\n", 30 - printed, | ||||
| 			opt[i].arg ? opt[i].arg : "", opt[i].help); | ||||
| 	} | ||||
| } | ||||
|   | ||||
							
								
								
									
										4
									
								
								opt.h
									
									
									
									
									
								
							
							
						
						
									
										4
									
								
								opt.h
									
									
									
									
									
								
							| @@ -11,7 +11,7 @@ | ||||
|  | ||||
| #include <stdlib.h> | ||||
|  | ||||
| struct options {  | ||||
| struct options { | ||||
| 	char shortopt; | ||||
| 	char *longopt; | ||||
| 	char *arg; | ||||
| @@ -23,6 +23,6 @@ void opt_init(int *optind); | ||||
| char opt_parse(int argc, char **argv, int *optind, char **optarg, | ||||
| 	       struct options *opt); | ||||
|  | ||||
| void opt_help(struct options *opt, FILE *out); | ||||
| void opt_help(struct options *opt, FILE * out); | ||||
|  | ||||
| #endif | ||||
|   | ||||
							
								
								
									
										301
									
								
								ue9.c
									
									
									
									
									
								
							
							
						
						
									
										301
									
								
								ue9.c
									
									
									
									
									
								
							| @@ -25,10 +25,10 @@ | ||||
| #include "util.h" | ||||
| #include "netutil.h" | ||||
|  | ||||
| #define UE9_TIMEOUT 5   /* Timeout for connect/send/recv, in seconds */ | ||||
| #define UE9_TIMEOUT 5		/* Timeout for connect/send/recv, in seconds */ | ||||
|  | ||||
| /* Fill checksums in data buffers, with "normal" checksum format */ | ||||
| void ue9_checksum_normal(uint8_t *buffer, size_t len) | ||||
| void ue9_checksum_normal(uint8_t * buffer, size_t len) | ||||
| { | ||||
| 	uint16_t sum = 0; | ||||
|  | ||||
| @@ -37,7 +37,7 @@ void ue9_checksum_normal(uint8_t *buffer, size_t len) | ||||
| 		exit(1); | ||||
| 	} | ||||
|  | ||||
| 	while (--len >= 1)  | ||||
| 	while (--len >= 1) | ||||
| 		sum += (uint16_t) buffer[len]; | ||||
| 	sum = (sum / 256) + (sum % 256); | ||||
| 	sum = (sum / 256) + (sum % 256); | ||||
| @@ -45,7 +45,7 @@ void ue9_checksum_normal(uint8_t *buffer, size_t len) | ||||
| } | ||||
|  | ||||
| /* Fill checksums in data buffers, with "extended" checksum format */ | ||||
| void ue9_checksum_extended(uint8_t *buffer, size_t len) | ||||
| void ue9_checksum_extended(uint8_t * buffer, size_t len) | ||||
| { | ||||
| 	uint16_t sum = 0; | ||||
|  | ||||
| @@ -65,7 +65,7 @@ void ue9_checksum_extended(uint8_t *buffer, size_t len) | ||||
| } | ||||
|  | ||||
| /* Verify checksums in data buffers, with "normal" checksum format. */ | ||||
| int ue9_verify_normal(uint8_t *buffer, size_t len) | ||||
| int ue9_verify_normal(uint8_t * buffer, size_t len) | ||||
| { | ||||
| 	uint8_t saved, new; | ||||
|  | ||||
| @@ -80,8 +80,7 @@ int ue9_verify_normal(uint8_t *buffer, size_t len) | ||||
| 	buffer[0] = saved; | ||||
|  | ||||
| 	if (new != saved) { | ||||
| 		verb("got %02x, expected %02x\n",  | ||||
| 		     saved, new); | ||||
| 		verb("got %02x, expected %02x\n", saved, new); | ||||
| 		return 0; | ||||
| 	} | ||||
|  | ||||
| @@ -89,7 +88,7 @@ int ue9_verify_normal(uint8_t *buffer, size_t len) | ||||
| } | ||||
|  | ||||
| /* Verify checksums in data buffers, with "extended" checksum format. */ | ||||
| int ue9_verify_extended(uint8_t *buffer, size_t len) | ||||
| int ue9_verify_extended(uint8_t * buffer, size_t len) | ||||
| { | ||||
| 	uint8_t saved[3], new[3]; | ||||
|  | ||||
| @@ -109,20 +108,19 @@ int ue9_verify_extended(uint8_t *buffer, size_t len) | ||||
| 	buffer[4] = saved[1]; | ||||
| 	buffer[5] = saved[2]; | ||||
|  | ||||
| 	if (saved[0] != new[0] ||  | ||||
| 	    saved[1] != new[1] || | ||||
| 	    saved[2] != new[2]) { | ||||
| 	if (saved[0] != new[0] || saved[1] != new[1] || saved[2] != new[2]) { | ||||
| 		verb("got %02x %02x %02x, expected %02x %02x %02x\n", | ||||
| 		     saved[0], saved[1], saved[2], new[0], new[1], new[2]); | ||||
| 		return 0; | ||||
| 	} | ||||
|  | ||||
| 	return 1; | ||||
| }		 | ||||
| } | ||||
|  | ||||
| /* Data conversion.  If calib is NULL, use uncalibrated conversions. */ | ||||
| double ue9_binary_to_analog(struct ue9Calibration *calib,  | ||||
| 			    uint8_t gain, uint8_t resolution, uint16_t data) | ||||
| double | ||||
| ue9_binary_to_analog(struct ue9Calibration *calib, | ||||
| 		     uint8_t gain, uint8_t resolution, uint16_t data) | ||||
| { | ||||
| 	double slope = 0, offset; | ||||
|  | ||||
| @@ -157,7 +155,7 @@ double ue9_binary_to_analog(struct ue9Calibration *calib, | ||||
| 		fprintf(stderr, "ue9_binary_to_analog: bad gain\n"); | ||||
| 		exit(1); | ||||
| 	} | ||||
| 	 | ||||
|  | ||||
| 	return data * slope + offset; | ||||
| } | ||||
|  | ||||
| @@ -165,7 +163,7 @@ double ue9_binary_to_analog(struct ue9Calibration *calib, | ||||
|    checksums on the outgoing packets, and verifies them on the | ||||
|    incoming packets.  Data in "out" is transmitted, data in "in" is | ||||
|    received. */ | ||||
| int ue9_command(int fd, uint8_t *out, uint8_t *in, int inlen) | ||||
| int ue9_command(int fd, uint8_t * out, uint8_t * in, int inlen) | ||||
| { | ||||
| 	int extended = 0, outlen; | ||||
| 	uint8_t saved_1, saved_3; | ||||
| @@ -184,8 +182,8 @@ int ue9_command(int fd, uint8_t *out, uint8_t *in, int inlen) | ||||
| 	} | ||||
|  | ||||
| 	/* Send request */ | ||||
| 	ret = send_all_timeout(fd, out, outlen, 0,  | ||||
| 			       & (struct timeval) { .tv_sec = UE9_TIMEOUT }); | ||||
| 	ret = send_all_timeout(fd, out, outlen, 0, &(struct timeval) { | ||||
| 			       .tv_sec = UE9_TIMEOUT}); | ||||
| 	if (ret < 0 || ret != outlen) { | ||||
| 		verb("short send %d\n", (int)ret); | ||||
| 		return -1; | ||||
| @@ -194,12 +192,12 @@ int ue9_command(int fd, uint8_t *out, uint8_t *in, int inlen) | ||||
| 	/* Save a few bytes that we'll want to compare against later, | ||||
| 	   in case the caller passed the same buffer twice. */ | ||||
| 	saved_1 = out[1]; | ||||
| 	if (extended)  | ||||
| 	if (extended) | ||||
| 		saved_3 = out[3]; | ||||
|  | ||||
| 	/* Receive result */ | ||||
| 	ret = recv_all_timeout(fd, in, inlen, 0, | ||||
| 			       & (struct timeval) { .tv_sec = UE9_TIMEOUT }); | ||||
| 	ret = recv_all_timeout(fd, in, inlen, 0, &(struct timeval) { | ||||
| 			       .tv_sec = UE9_TIMEOUT}); | ||||
| 	if (ret < 0 || ret != inlen) { | ||||
| 		verb("short recv %d\n", (int)ret); | ||||
| 		return -1; | ||||
| @@ -214,24 +212,23 @@ int ue9_command(int fd, uint8_t *out, uint8_t *in, int inlen) | ||||
| 		verb("returned extended data is the wrong len\n"); | ||||
| 	else if (!extended && (inlen != (2 + (in[1] & 7) * 2))) | ||||
| 		verb("returned data is the wrong len\n"); | ||||
| 	else if (extended && !ue9_verify_extended(in, inlen))  | ||||
| 	else if (extended && !ue9_verify_extended(in, inlen)) | ||||
| 		verb("extended checksum is invalid\n"); | ||||
| 	else if (!ue9_verify_normal(in, extended ? 6 : inlen)) | ||||
| 		verb("normal checksum is invalid\n"); | ||||
| 	else | ||||
| 		return 0;  /* looks good */ | ||||
| 		return 0;	/* looks good */ | ||||
|  | ||||
| 	return -1; | ||||
| } | ||||
|  | ||||
|  | ||||
| /* Read a memory block from the device.  Returns -1 on error. */ | ||||
| int ue9_memory_read(int fd, int blocknum, uint8_t *buffer, int len)  | ||||
| int ue9_memory_read(int fd, int blocknum, uint8_t * buffer, int len) | ||||
| { | ||||
| 	uint8_t sendbuf[8], recvbuf[136]; | ||||
|  | ||||
| 	if (len != 128) { | ||||
| 		fprintf(stderr,"ue9_memory_read: buffer length must be 128\n"); | ||||
| 		fprintf(stderr, "ue9_memory_read: buffer length must be 128\n"); | ||||
| 		exit(1); | ||||
| 	} | ||||
|  | ||||
| @@ -250,11 +247,11 @@ int ue9_memory_read(int fd, int blocknum, uint8_t *buffer, int len) | ||||
| 	/* Got it */ | ||||
| 	memcpy(buffer, recvbuf + 8, len); | ||||
|  | ||||
| 	return 0;		 | ||||
| 	return 0; | ||||
| } | ||||
|  | ||||
| /* Convert 64-bit fixed point to double type */ | ||||
| double ue9_fp64_to_double(uint8_t *data) | ||||
| double ue9_fp64_to_double(uint8_t * data) | ||||
| { | ||||
| 	int32_t a; | ||||
| 	uint32_t b; | ||||
| @@ -271,43 +268,48 @@ int ue9_get_calibration(int fd, struct ue9Calibration *calib) | ||||
| 	uint8_t buf[128]; | ||||
|  | ||||
| 	/* Block 0 */ | ||||
| 	if (ue9_memory_read(fd, 0, buf, 128) < 0) return -1; | ||||
| 	calib->unipolarSlope[0]    = ue9_fp64_to_double(buf + 0); | ||||
| 	calib->unipolarOffset[0]   = ue9_fp64_to_double(buf + 8); | ||||
| 	calib->unipolarSlope[1]    = ue9_fp64_to_double(buf + 16); | ||||
| 	calib->unipolarOffset[1]   = ue9_fp64_to_double(buf + 24); | ||||
| 	calib->unipolarSlope[2]    = ue9_fp64_to_double(buf + 32); | ||||
| 	calib->unipolarOffset[2]   = ue9_fp64_to_double(buf + 40); | ||||
| 	calib->unipolarSlope[3]    = ue9_fp64_to_double(buf + 48); | ||||
| 	calib->unipolarOffset[3]   = ue9_fp64_to_double(buf + 56); | ||||
| 	if (ue9_memory_read(fd, 0, buf, 128) < 0) | ||||
| 		return -1; | ||||
| 	calib->unipolarSlope[0] = ue9_fp64_to_double(buf + 0); | ||||
| 	calib->unipolarOffset[0] = ue9_fp64_to_double(buf + 8); | ||||
| 	calib->unipolarSlope[1] = ue9_fp64_to_double(buf + 16); | ||||
| 	calib->unipolarOffset[1] = ue9_fp64_to_double(buf + 24); | ||||
| 	calib->unipolarSlope[2] = ue9_fp64_to_double(buf + 32); | ||||
| 	calib->unipolarOffset[2] = ue9_fp64_to_double(buf + 40); | ||||
| 	calib->unipolarSlope[3] = ue9_fp64_to_double(buf + 48); | ||||
| 	calib->unipolarOffset[3] = ue9_fp64_to_double(buf + 56); | ||||
|  | ||||
| 	/* Block 1 */ | ||||
| 	if (ue9_memory_read(fd, 1, buf, 128) < 0) return -1; | ||||
| 	calib->bipolarSlope        = ue9_fp64_to_double(buf + 0); | ||||
| 	calib->bipolarOffset       = ue9_fp64_to_double(buf + 8); | ||||
| 	if (ue9_memory_read(fd, 1, buf, 128) < 0) | ||||
| 		return -1; | ||||
| 	calib->bipolarSlope = ue9_fp64_to_double(buf + 0); | ||||
| 	calib->bipolarOffset = ue9_fp64_to_double(buf + 8); | ||||
|  | ||||
| 	/* Block 2 */ | ||||
| 	if (ue9_memory_read(fd, 2, buf, 128) < 0) return -1; | ||||
| 	calib->DACSlope[0]         = ue9_fp64_to_double(buf + 0); | ||||
| 	calib->DACOffset[0]        = ue9_fp64_to_double(buf + 8); | ||||
| 	calib->DACSlope[1]         = ue9_fp64_to_double(buf + 16); | ||||
| 	calib->DACOffset[1]        = ue9_fp64_to_double(buf + 24); | ||||
| 	calib->tempSlope           = ue9_fp64_to_double(buf + 32); | ||||
| 	calib->tempSlopeLow        = ue9_fp64_to_double(buf + 48); | ||||
| 	calib->calTemp             = ue9_fp64_to_double(buf + 64); | ||||
| 	calib->Vref                = ue9_fp64_to_double(buf + 72); | ||||
| 	calib->VrefDiv2            = ue9_fp64_to_double(buf + 88); | ||||
| 	calib->VsSlope             = ue9_fp64_to_double(buf + 96); | ||||
| 	if (ue9_memory_read(fd, 2, buf, 128) < 0) | ||||
| 		return -1; | ||||
| 	calib->DACSlope[0] = ue9_fp64_to_double(buf + 0); | ||||
| 	calib->DACOffset[0] = ue9_fp64_to_double(buf + 8); | ||||
| 	calib->DACSlope[1] = ue9_fp64_to_double(buf + 16); | ||||
| 	calib->DACOffset[1] = ue9_fp64_to_double(buf + 24); | ||||
| 	calib->tempSlope = ue9_fp64_to_double(buf + 32); | ||||
| 	calib->tempSlopeLow = ue9_fp64_to_double(buf + 48); | ||||
| 	calib->calTemp = ue9_fp64_to_double(buf + 64); | ||||
| 	calib->Vref = ue9_fp64_to_double(buf + 72); | ||||
| 	calib->VrefDiv2 = ue9_fp64_to_double(buf + 88); | ||||
| 	calib->VsSlope = ue9_fp64_to_double(buf + 96); | ||||
|  | ||||
| 	/* Block 3 */ | ||||
| 	if (ue9_memory_read(fd, 3, buf, 128) < 0) return -1; | ||||
| 	calib->hiResUnipolarSlope  = ue9_fp64_to_double(buf + 0); | ||||
| 	if (ue9_memory_read(fd, 3, buf, 128) < 0) | ||||
| 		return -1; | ||||
| 	calib->hiResUnipolarSlope = ue9_fp64_to_double(buf + 0); | ||||
| 	calib->hiResUnipolarOffset = ue9_fp64_to_double(buf + 8); | ||||
|  | ||||
| 	/* Block 4 */ | ||||
| 	if (ue9_memory_read(fd, 4, buf, 128) < 0) return -1; | ||||
| 	calib->hiResBipolarSlope   = ue9_fp64_to_double(buf + 0); | ||||
| 	calib->hiResBipolarOffset  = ue9_fp64_to_double(buf + 8); | ||||
| 	if (ue9_memory_read(fd, 4, buf, 128) < 0) | ||||
| 		return -1; | ||||
| 	calib->hiResBipolarSlope = ue9_fp64_to_double(buf + 0); | ||||
| 	calib->hiResBipolarOffset = ue9_fp64_to_double(buf + 8); | ||||
|  | ||||
| 	/* All done */ | ||||
| 	return 1; | ||||
| @@ -318,7 +320,7 @@ int ue9_get_comm_config(int fd, struct ue9CommConfig *config) | ||||
| { | ||||
| 	uint8_t sendbuf[18]; | ||||
| 	uint8_t recvbuf[24]; | ||||
| 	 | ||||
|  | ||||
| 	memset(sendbuf, 0, sizeof(sendbuf)); | ||||
| 	memset(config, 0, sizeof(struct ue9CommConfig)); | ||||
|  | ||||
| @@ -338,7 +340,7 @@ int ue9_get_control_config(int fd, struct ue9ControlConfig *config) | ||||
| { | ||||
| 	uint8_t sendbuf[18]; | ||||
| 	uint8_t recvbuf[24]; | ||||
| 	 | ||||
|  | ||||
| 	memset(sendbuf, 0, sizeof(sendbuf)); | ||||
| 	memset(config, 0, sizeof(struct ue9ControlConfig)); | ||||
|  | ||||
| @@ -377,7 +379,7 @@ int ue9_open(const char *host, int port) | ||||
| 	} | ||||
|  | ||||
| 	/* Set initial window size hint to workaround LabJack firmware bug */ | ||||
| 	setsockopt(fd, SOL_SOCKET, SO_SNDBUF, (void *)&window_size,  | ||||
| 	setsockopt(fd, SOL_SOCKET, SO_SNDBUF, (void *)&window_size, | ||||
| 		   sizeof(window_size)); | ||||
| 	setsockopt(fd, SOL_SOCKET, SO_RCVBUF, (void *)&window_size, | ||||
| 		   sizeof(window_size)); | ||||
| @@ -390,13 +392,14 @@ int ue9_open(const char *host, int port) | ||||
| 		verb("gethostbyname(\"%s\") failed\n", host); | ||||
| 		return -1; | ||||
| 	} | ||||
| 	address.sin_addr = *((struct in_addr *) he->h_addr); | ||||
| 	address.sin_addr = *((struct in_addr *)he->h_addr); | ||||
|  | ||||
| 	debug("Resolved %s -> %s\n", host, inet_ntoa(address.sin_addr)); | ||||
|  | ||||
| 	/* Connect */ | ||||
| 	if (connect_timeout(fd, (struct sockaddr *) &address, sizeof(address), | ||||
| 			    & (struct timeval) { .tv_sec = UE9_TIMEOUT }) < 0) { | ||||
| 	if (connect_timeout(fd, (struct sockaddr *)&address, sizeof(address), | ||||
| 			    &(struct timeval) { | ||||
| 			    .tv_sec = UE9_TIMEOUT}) < 0) { | ||||
| 		verb("connection to %s:%d failed: %s\n", | ||||
| 		     inet_ntoa(address.sin_addr), port, compat_strerror(errno)); | ||||
| 		return -1; | ||||
| @@ -409,7 +412,7 @@ int ue9_open(const char *host, int port) | ||||
| void ue9_close(int fd) | ||||
| { | ||||
| 	/* does anyone actually call shutdown these days? */ | ||||
| 	shutdown(fd, 2 /* SHUT_RDWR */); | ||||
| 	shutdown(fd, 2 /* SHUT_RDWR */ ); | ||||
| 	close(fd); | ||||
| } | ||||
|  | ||||
| @@ -423,12 +426,20 @@ double ue9_compute_rate(uint8_t scanconfig, uint16_t scaninterval) | ||||
| 	   Channels are scanned as quickly as possible. */ | ||||
|  | ||||
| 	switch ((scanconfig >> 3) & 3) { | ||||
| 	case 0: clock = 4e6; break; | ||||
| 	case 1: clock = 48e6; break; | ||||
| 	case 2: clock = 750e3; break; | ||||
| 	case 3: clock = 24e6; break; | ||||
| 	case 0: | ||||
| 		clock = 4e6; | ||||
| 		break; | ||||
| 	case 1: | ||||
| 		clock = 48e6; | ||||
| 		break; | ||||
| 	case 2: | ||||
| 		clock = 750e3; | ||||
| 		break; | ||||
| 	case 3: | ||||
| 		clock = 24e6; | ||||
| 		break; | ||||
| 	} | ||||
| 	 | ||||
|  | ||||
| 	if (scanconfig & 0x2) | ||||
| 		clock /= 256; | ||||
|  | ||||
| @@ -440,20 +451,25 @@ double ue9_compute_rate(uint8_t scanconfig, uint16_t scaninterval) | ||||
|  | ||||
| /* Choose the best ScanConfig and ScanInterval parameters for the | ||||
|    desired scanrate.  Returns -1 if no valid config found */ | ||||
| int ue9_choose_scan(double desired_rate, double *actual_rate, | ||||
| 		    uint8_t *scanconfig, uint16_t *scaninterval) | ||||
| int | ||||
| ue9_choose_scan(double desired_rate, double *actual_rate, | ||||
| 		uint8_t * scanconfig, uint16_t * scaninterval) | ||||
| { | ||||
| 	int i; | ||||
| 	struct { double clock; uint8_t config; } valid[] = { | ||||
| 		{ 48e6,        0x08 }, | ||||
| 		{ 24e6,        0x18 }, | ||||
| 		{ 4e6,         0x00 }, | ||||
| 		{ 750e3,       0x10 }, | ||||
| 		{ 48e6 / 256,  0x0a }, | ||||
| 		{ 24e6 / 256,  0x1a }, | ||||
| 		{ 4e6 / 256,   0x02 }, | ||||
| 		{ 750e3 / 256, 0x12 }, | ||||
| 		{ 0, 0 } }; | ||||
| 	struct { | ||||
| 		double clock; | ||||
| 		uint8_t config; | ||||
| 	} valid[] = { | ||||
| 		{ | ||||
| 		48e6, 0x08}, { | ||||
| 		24e6, 0x18}, { | ||||
| 		4e6, 0x00}, { | ||||
| 		750e3, 0x10}, { | ||||
| 		48e6 / 256, 0x0a}, { | ||||
| 		24e6 / 256, 0x1a}, { | ||||
| 		4e6 / 256, 0x02}, { | ||||
| 		750e3 / 256, 0x12}, { | ||||
| 	0, 0}}; | ||||
|  | ||||
| 	/* Start with the fastest clock frequency.  If the | ||||
| 	   scaninterval would be too large, knock it down until it | ||||
| @@ -461,7 +477,7 @@ int ue9_choose_scan(double desired_rate, double *actual_rate, | ||||
| 	for (i = 0; valid[i].clock != 0; i++) { | ||||
| 		double interval = valid[i].clock / desired_rate; | ||||
|  | ||||
| 		debug("Considering clock %lf (interval %lf)\n",  | ||||
| 		debug("Considering clock %lf (interval %lf)\n", | ||||
| 		      valid[i].clock, interval); | ||||
|  | ||||
| 		if (interval >= 0.5 && interval < 65535.5) { | ||||
| @@ -469,14 +485,14 @@ int ue9_choose_scan(double desired_rate, double *actual_rate, | ||||
| 			*scaninterval = floor(interval + 0.5); | ||||
|  | ||||
| 			*scanconfig = valid[i].config; | ||||
| 			*actual_rate = ue9_compute_rate( | ||||
| 				*scanconfig, *scaninterval); | ||||
| 			*actual_rate = | ||||
| 			    ue9_compute_rate(*scanconfig, *scaninterval); | ||||
|  | ||||
| 			debug("Config 0x%02x, desired %lf, actual %lf\n",  | ||||
| 			debug("Config 0x%02x, desired %lf, actual %lf\n", | ||||
| 			      *scanconfig, desired_rate, *actual_rate); | ||||
|  | ||||
| 			return 0; | ||||
| 		}		 | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	return -1; | ||||
| @@ -486,8 +502,8 @@ int ue9_choose_scan(double desired_rate, double *actual_rate, | ||||
| void ue9_buffer_flush(int fd) | ||||
| { | ||||
| 	uint8_t sendbuf[2], recvbuf[2]; | ||||
| 	 | ||||
| 	sendbuf[1] = 0x08; /* FlushBuffer */ | ||||
|  | ||||
| 	sendbuf[1] = 0x08;	/* FlushBuffer */ | ||||
|  | ||||
| 	if (ue9_command(fd, sendbuf, recvbuf, sizeof(recvbuf)) < 0) { | ||||
| 		verb("command failed\n"); | ||||
| @@ -498,14 +514,14 @@ void ue9_buffer_flush(int fd) | ||||
| int ue9_stream_stop(int fd) | ||||
| { | ||||
| 	uint8_t sendbuf[2], recvbuf[4]; | ||||
| 	 | ||||
|  | ||||
| 	sendbuf[1] = 0xB0; | ||||
|  | ||||
| 	if (ue9_command(fd, sendbuf, recvbuf, sizeof(recvbuf)) < 0) { | ||||
| 		verb("command failed\n"); | ||||
| 		return -1; | ||||
| 	} | ||||
| 	 | ||||
|  | ||||
| 	if (recvbuf[2] == STREAM_NOT_RUNNING || recvbuf[2] == 0) | ||||
| 		return 0; | ||||
|  | ||||
| @@ -517,14 +533,14 @@ int ue9_stream_stop(int fd) | ||||
| int ue9_stream_start(int fd) | ||||
| { | ||||
| 	uint8_t sendbuf[2], recvbuf[4]; | ||||
| 	 | ||||
|  | ||||
| 	sendbuf[1] = 0xA8; | ||||
|  | ||||
| 	if (ue9_command(fd, sendbuf, recvbuf, sizeof(recvbuf)) < 0) { | ||||
| 		verb("command failed\n"); | ||||
| 		return -1; | ||||
| 	} | ||||
| 	 | ||||
|  | ||||
| 	if (recvbuf[2] == 0) | ||||
| 		return 0; | ||||
|  | ||||
| @@ -534,27 +550,27 @@ int ue9_stream_start(int fd) | ||||
|  | ||||
| /* "Simple" stream configuration, assumes the channels are all  | ||||
|    configured with the same gain. */ | ||||
| int ue9_streamconfig_simple(int fd, int *channel_list, int channel_count, | ||||
| 			    uint8_t scanconfig, uint16_t scaninterval, | ||||
| 			    uint8_t gain) | ||||
| int | ||||
| ue9_streamconfig_simple(int fd, int *channel_list, int channel_count, | ||||
| 			uint8_t scanconfig, uint16_t scaninterval, uint8_t gain) | ||||
| { | ||||
| 	int i; | ||||
| 	uint8_t buf[256]; | ||||
|  | ||||
| 	/* Set up StreamConfig command with channels and scan options */ | ||||
| 	buf[1] = 0xF8;           	/* Extended command */ | ||||
| 	buf[2] = channel_count + 3;   	/* Command data words */ | ||||
| 	buf[3] = 0x11;           	/* StreamConfig */ | ||||
| 	buf[6] = channel_count;		/* Number of channels */ | ||||
| 	buf[7] = 12;        		/* Bit resolution */ | ||||
| 	buf[8] = 0;			/* Extra settling time */ | ||||
| 	buf[1] = 0xF8;		/* Extended command */ | ||||
| 	buf[2] = channel_count + 3;	/* Command data words */ | ||||
| 	buf[3] = 0x11;		/* StreamConfig */ | ||||
| 	buf[6] = channel_count;	/* Number of channels */ | ||||
| 	buf[7] = 12;		/* Bit resolution */ | ||||
| 	buf[8] = 0;		/* Extra settling time */ | ||||
| 	buf[9] = scanconfig; | ||||
| 	buf[10] = scaninterval & 0xff; | ||||
| 	buf[11] = scaninterval >> 8; | ||||
| 	 | ||||
|  | ||||
| 	for (i = 0; i < channel_count; i++) { | ||||
| 		buf[12 + 2*i] = channel_list[i];    /* Channel number */ | ||||
| 		buf[13 + 2*i] = gain;		    /* Gain/bipolar setup */ | ||||
| 		buf[12 + 2 * i] = channel_list[i];	/* Channel number */ | ||||
| 		buf[13 + 2 * i] = gain;	/* Gain/bipolar setup */ | ||||
| 	} | ||||
|  | ||||
| 	/* Send StreamConfig */ | ||||
| @@ -562,7 +578,7 @@ int ue9_streamconfig_simple(int fd, int *channel_list, int channel_count, | ||||
| 		debug("command failed\n"); | ||||
| 		return -1; | ||||
| 	} | ||||
| 	 | ||||
|  | ||||
| 	if (buf[6] != 0) { | ||||
| 		verb("returned error %s\n", ue9_error(buf[6])); | ||||
| 		return -1; | ||||
| @@ -571,10 +587,58 @@ int ue9_streamconfig_simple(int fd, int *channel_list, int channel_count, | ||||
| 	return 0; | ||||
| } | ||||
|  | ||||
| /* Timer configuration */ | ||||
| int ue9_timer_config(int fd, int *mode_list, int mode_count, int divisor) | ||||
| { | ||||
| 	int i; | ||||
| 	uint8_t buf[256]; | ||||
|  | ||||
| 	if (mode_count < 0 || mode_count > 6) { | ||||
| 		verb("invalid count\n"); | ||||
| 		return -1; | ||||
| 	} | ||||
|  | ||||
| 	/* Set up TimerConfig command */ | ||||
| 	buf[1] = 0xF8;		/* Extended command */ | ||||
| 	buf[2] = 0x0C;		/* Command data words */ | ||||
| 	buf[3] = 0x18;		/* TimerConfig */ | ||||
| 	buf[6] = divisor;	/* TimerClockDivisor */ | ||||
| 	buf[7] = 0x80 | mode_count;	/* Number of timers enabled, UpdateConfig=1 */ | ||||
| 	buf[8] = 0x01;		/* TimerClockBase = System 48MHz */ | ||||
| 	buf[9] = 0x00;		/* Don't reset */ | ||||
|  | ||||
| 	for (i = 0; i < 6; i++) { | ||||
| 		if (i < mode_count) | ||||
| 			buf[10 + 3 * i] = mode_list[i]; | ||||
| 		else | ||||
| 			buf[10 + 3 * i] = 0; | ||||
| 		buf[11 + 3 * i] = 0; | ||||
| 		buf[12 + 3 * i] = 0; | ||||
| 	} | ||||
|  | ||||
| 	buf[28] = 0; | ||||
| 	buf[29] = 0; | ||||
|  | ||||
| 	/* Send StreamConfig */ | ||||
| 	if (ue9_command(fd, buf, buf, 40) < 0) { | ||||
| 		debug("command failed\n"); | ||||
| 		return -1; | ||||
| 	} | ||||
|  | ||||
| 	if (buf[6] != 0) { | ||||
| 		verb("returned error %s\n", ue9_error(buf[6])); | ||||
| 		return -1; | ||||
| 	} | ||||
|  | ||||
| 	debug("timer EnableStatus=0x%02x\n", buf[7]); | ||||
|  | ||||
| 	return 0; | ||||
| } | ||||
|  | ||||
| /* Stream data and pass it to the data callback.  If callback returns | ||||
|    negative, stops reading and returns 0.  Returns < 0 on error. */ | ||||
| int ue9_stream_data(int fd, int channels,  | ||||
| 		    ue9_stream_cb_t callback, void *context) | ||||
| int | ||||
| ue9_stream_data(int fd, int channels, ue9_stream_cb_t callback, void *context) | ||||
| { | ||||
| 	int ret; | ||||
| 	uint8_t buf[46]; | ||||
| @@ -582,11 +646,11 @@ int ue9_stream_data(int fd, int channels, | ||||
| 	int channel = 0; | ||||
| 	int i; | ||||
| 	uint16_t data[channels]; | ||||
| 	 | ||||
|  | ||||
| 	for (;;) { | ||||
| 		/* Receive data */ | ||||
| 		ret = recv_all_timeout(fd, buf, 46, 0, & (struct timeval)  | ||||
| 			{ .tv_sec = UE9_TIMEOUT }); | ||||
| 		ret = recv_all_timeout(fd, buf, 46, 0, &(struct timeval) { | ||||
| 				       .tv_sec = UE9_TIMEOUT}); | ||||
|  | ||||
| 		/* Verify packet format */ | ||||
| 		if (ret != 46) { | ||||
| @@ -594,8 +658,7 @@ int ue9_stream_data(int fd, int channels, | ||||
| 			return -1; | ||||
| 		} | ||||
|  | ||||
| 		if (!ue9_verify_extended(buf, 46) ||  | ||||
| 		    !ue9_verify_normal(buf, 6)) { | ||||
| 		if (!ue9_verify_extended(buf, 46) || !ue9_verify_normal(buf, 6)) { | ||||
| 			verb("bad checksum\n"); | ||||
| 			return -2; | ||||
| 		} | ||||
| @@ -617,13 +680,13 @@ int ue9_stream_data(int fd, int channels, | ||||
| 			return -5; | ||||
| 		} | ||||
| 		packet++; | ||||
| 		 | ||||
|  | ||||
| 		/* Check comm processor backlog (up to 512 kB) */ | ||||
| 		if (buf[45] & 0x80) { | ||||
| 			verb("buffer overflow in CommBacklog, aborting\n"); | ||||
| 			return -6; | ||||
| 		} | ||||
| 		if ((buf[45] & 0x7f) > 112)  | ||||
| 		if ((buf[45] & 0x7f) > 112) | ||||
| 			debug("warning: CommBacklog is high (%d bytes)\n", | ||||
| 			      (buf[45] & 0x7f) * 4096); | ||||
|  | ||||
| @@ -638,16 +701,22 @@ int ue9_stream_data(int fd, int channels, | ||||
|  | ||||
| 		/* Read samples from the buffer */ | ||||
| 		for (i = 12; i <= 42; i += 2) { | ||||
| 			data[channel++] = buf[i] + (buf[i+1] << 8); | ||||
| 			if (channel < channels)  | ||||
| 			data[channel++] = buf[i] + (buf[i + 1] << 8); | ||||
| 			if (channel < channels) | ||||
| 				continue; | ||||
|  | ||||
| 			/* Received a full scan, send to callback */ | ||||
| 			channel = 0; | ||||
| 			if ((*callback)(channels, data, context) < 0) { | ||||
| 			if ((*callback) (channels, data, context) < 0) { | ||||
| 				/* We're done */ | ||||
| 				return 0; | ||||
| 			} | ||||
| 		} | ||||
| 	} | ||||
| } | ||||
|  | ||||
| /* | ||||
| Local variables: | ||||
| c-basic-offset: 2 | ||||
| End: | ||||
| */ | ||||
|   | ||||
							
								
								
									
										29
									
								
								ue9.h
									
									
									
									
									
								
							
							
						
						
									
										29
									
								
								ue9.h
									
									
									
									
									
								
							| @@ -42,7 +42,7 @@ struct ue9CommConfig { | ||||
| 	in_addr_t address; | ||||
| 	in_addr_t gateway; | ||||
| 	in_addr_t subnet; | ||||
|         in_port_t portA; | ||||
| 	in_port_t portA; | ||||
| 	in_port_t portB; | ||||
| 	uint8_t dhcp_enabled; | ||||
| 	uint8_t product_id; | ||||
| @@ -74,25 +74,27 @@ struct ue9ControlConfig { | ||||
| #define UE9_UNIPOLAR_GAIN8 0x03 | ||||
| #define UE9_BIPOLAR_GAIN1 0x08 | ||||
|  | ||||
| #define UE9_CHANNELS 14 | ||||
| #define UE9_MAX_CHANNEL_COUNT 128 | ||||
| #define UE9_MAX_CHANNEL 255 | ||||
| #define UE9_TIMERS 6 | ||||
|  | ||||
| /* Fill checksums in data buffers */ | ||||
| void ue9_checksum_normal(uint8_t *buffer, size_t len); | ||||
| void ue9_checksum_extended(uint8_t *buffer, size_t len); | ||||
| void ue9_checksum_normal(uint8_t * buffer, size_t len); | ||||
| void ue9_checksum_extended(uint8_t * buffer, size_t len); | ||||
|  | ||||
| /* Verify checksums in data buffers.  Returns 0 on error. */ | ||||
| int ue9_verify_normal(uint8_t *buffer, size_t len); | ||||
| int ue9_verify_extended(uint8_t *buffer, size_t len); | ||||
| int ue9_verify_normal(uint8_t * buffer, size_t len); | ||||
| int ue9_verify_extended(uint8_t * buffer, size_t len); | ||||
|  | ||||
| /* Open/close TCP/IP connection to the UE9 */ | ||||
| int ue9_open(const char *host, int port); | ||||
| void ue9_close(int fd); | ||||
|  | ||||
| /* Read a memory block from the device.  Returns -1 on error. */ | ||||
| int ue9_memory_read(int fd, int blocknum, uint8_t *buffer, int len); | ||||
| int ue9_memory_read(int fd, int blocknum, uint8_t * buffer, int len); | ||||
|  | ||||
| /* Convert 64-bit fixed point to double type */ | ||||
| double ue9_fp64_to_double(uint8_t *data); | ||||
| double ue9_fp64_to_double(uint8_t * data); | ||||
|  | ||||
| /* Retrieve calibration data or configuration from the device */ | ||||
| int ue9_get_calibration(int fd, struct ue9Calibration *calib); | ||||
| @@ -100,7 +102,7 @@ int ue9_get_comm_config(int fd, struct ue9CommConfig *config); | ||||
| int ue9_get_control_config(int fd, struct ue9ControlConfig *config); | ||||
|  | ||||
| /* Data conversion.  If calib is NULL, use uncalibrated conversions. */ | ||||
| double ue9_binary_to_analog(struct ue9Calibration *calib,  | ||||
| double ue9_binary_to_analog(struct ue9Calibration *calib, | ||||
| 			    uint8_t gain, uint8_t resolution, uint16_t data); | ||||
|  | ||||
| /* Compute scanrate based on the provided values. */ | ||||
| @@ -109,7 +111,7 @@ double ue9_compute_rate(uint8_t scanconfig, uint16_t scaninterval); | ||||
| /* Choose the best ScanConfig and ScanInterval parameters for the | ||||
|    desired scanrate.  Returns 0 if nothing can be chosen. */ | ||||
| int ue9_choose_scan(double desired_rate, double *actual_rate, | ||||
| 		    uint8_t *scanconfig, uint16_t *scaninterval); | ||||
| 		    uint8_t * scanconfig, uint16_t * scaninterval); | ||||
|  | ||||
| /* Flush data buffers */ | ||||
| void ue9_buffer_flush(int fd); | ||||
| @@ -124,7 +126,7 @@ int ue9_stream_start(int fd); | ||||
|    checksums on the outgoing packets, and verifies them on the | ||||
|    incoming packets.  Data in "out" is transmitted, data in "in" is | ||||
|    received. */ | ||||
| int ue9_command(int fd, uint8_t *out, uint8_t *in, int inlen); | ||||
| int ue9_command(int fd, uint8_t * out, uint8_t * in, int inlen); | ||||
|  | ||||
| /* "Simple" stream configuration, assumes the channels are all  | ||||
|    configured with the same gain. */ | ||||
| @@ -132,9 +134,12 @@ int ue9_streamconfig_simple(int fd, int *channel_list, int channel_count, | ||||
| 			    uint8_t scanconfig, uint16_t scaninterval, | ||||
| 			    uint8_t gain); | ||||
|  | ||||
| /* Timer configuration */ | ||||
| int ue9_timer_config(int fd, int *mode_list, int mode_count, int divisor); | ||||
|  | ||||
| /* Stream data and pass it to the data callback.  If callback returns | ||||
|    negative, stops reading and returns 0.  Returns < 0 on error. */ | ||||
| typedef int (*ue9_stream_cb_t)(int channels, uint16_t *data, void *context); | ||||
| typedef int (*ue9_stream_cb_t) (int channels, uint16_t * data, void *context); | ||||
| int ue9_stream_data(int fd, int channels, | ||||
| 		    ue9_stream_cb_t callback, void *context); | ||||
|  | ||||
|   | ||||
		Reference in New Issue
	
	Block a user