icsneoc2 Examples

Simple

Download example

#include <icsneo/icsneoc2.h>
#include <icsneo/icsneoc2messages.h>
#include <icsneo/icsneoc2settings.h>

#ifdef _WIN32
#include <windows.h>
#else
#include <unistd.h>
#endif

#include <stdio.h>
#include <inttypes.h>
#include <time.h>

/**
 * Sleeps for a specified number of milliseconds using Sleep() on Windows and sleep() on *nix.
 *
 * @param ms The number of milliseconds to sleep.
 */
void sleep_ms(uint32_t ms) {
#ifdef _WIN32
	Sleep(ms);
#else
	sleep(ms / 1000);
#endif
}

/**
 * Prints an error message with the given string and error code.
 *
 * If the error code is not icsneoc2_error_success, prints the error string for the given error code
 * and returns the error code.
 *
 * @param message The message to print.
 * @param error The error code to print.
 * @return error as int
 */
int print_error_code(const char* message, icsneoc2_error_t error) {
	char error_str[64];
	size_t error_str_len = sizeof(error_str);
	icsneoc2_error_t res = icsneoc2_error_code_get(error, error_str, &error_str_len);
	if(res != icsneoc2_error_success) {
		printf("%s: Failed to get string for error code %d with error code %d\n", message, error, res);
		return res;
	}
	printf("%s: \"%s\" (%u)\n", message, error_str, error);
	return (int)error;
}

/**
 * Processes a list of messages from a device.
 *
 * This function iterates over a given array of messages received from a specified device.
 * For each message in the array, it retrieves and prints the message type and bus type.
 * If an error occurs while retrieving these details, an error message is printed.
 *
 * @param messages An array of pointers to icsneoc2_message_t structures containing the messages to process.
 * @param messages_count The number of messages in the messages array.
 *
 * @return An icsneoc2_error_t value indicating success or failure of the message processing.
 */
int process_messages(icsneoc2_message_t** messages, size_t messages_count);

/**
 * Prints all events
 *
 * @param device_description A description of the device used in the output.
 */
void print_events(const char* device_description);

/**
 * Transmits a series of CAN messages from a device.
 *
 * This function creates and transmits 100 CAN messages with incrementing payload data.
 * Each message is configured with specific attributes such as network ID, arbitration
 * ID, CANFD status, extended status, and baudrate switch. After successfully transmitting
 * each message, it is freed from memory.
 *
 * @param device A pointer to the icsneoc2_device_t structure representing the device to transmit messages from.
 *
 * @return An icsneoc2_error_t value indicating success or failure of the message transmission process.
 */
int transmit_can_messages(icsneoc2_device_t* device);

/**
 * Get the RTC (Real time clock) of a device and print it.
 *
 * @param[in] device The device to get the RTC of.
 * @param[in] description A description of the device for printing purpose.
 *
 * @return icsneoc2_error_t icsneoc2_error_success if successful, icsneoc2_error_invalid_parameters otherwise.
 */
icsneoc2_error_t get_and_print_rtc(icsneoc2_device_t* device);

int main() {
	icsneoc2_device_info_t* found_devices = NULL;

	printf("Finding devices...\n");
	icsneoc2_error_t res = icsneoc2_device_enumerate(0, &found_devices);
	if(res != icsneoc2_error_success) {
		return print_error_code("\tFailed to find devices", res);
	}
	if(found_devices == NULL) {
		printf("No devices found, exiting\n");
		return 0;
	}
	// Count and list off the devices
	size_t devices_count = 0;
	for(icsneoc2_device_info_t* cur = found_devices; cur != NULL; cur = icsneoc2_device_info_next(cur)) {
		devices_count++;
	}
	printf("OK, %zu device%s found\n", devices_count, devices_count == 1 ? "" : "s");
	for(icsneoc2_device_info_t* cur = found_devices; cur != NULL; cur = icsneoc2_device_info_next(cur)) {
		// Get description of the device
		char description[255] = {0};
		size_t description_length = 255;
		res = icsneoc2_device_info_description_get(cur, description, &description_length);
		if(res != icsneoc2_error_success) {
			icsneoc2_enumeration_free(found_devices);
			return print_error_code("\tFailed to get device description", res);
		};
		printf("%.*s\n", (int)description_length, description);
		// Open the device without RTC sync and going online
		icsneoc2_open_options_t options = icsneoc2_open_options_default;
		options &= ~ICSNEOC2_OPEN_OPTIONS_SYNC_RTC;
		options &= ~ICSNEOC2_OPEN_OPTIONS_GO_ONLINE;
		printf("\tDevice open options: 0x%x\n", options);
		printf("\tOpening device: %s...\n", description);
		icsneoc2_device_t* open_device = NULL;
		res = icsneoc2_device_create(cur, &open_device);
		if(res != icsneoc2_error_success) {
			icsneoc2_enumeration_free(found_devices);
			return print_error_code("\tFailed to create device", res);
		}
		res = icsneoc2_device_open(open_device, options);
		if(res != icsneoc2_error_success) {
			icsneoc2_device_free(open_device);
			icsneoc2_enumeration_free(found_devices);
			return print_error_code("\tFailed to open device", res);
		};

		// Get timestamp resolution of the device
		printf("\tGetting timestamp resolution... ");
		uint32_t timestamp_resolution = 0;
		res = icsneoc2_device_timestamp_resolution_get(open_device, &timestamp_resolution);
		if(res != icsneoc2_error_success) {
			print_events(description);
			icsneoc2_device_free(open_device);
			icsneoc2_enumeration_free(found_devices);
			return print_error_code("\tFailed to get timestamp resolution", res);
		}
		printf("%uns\n", timestamp_resolution);
		// Get baudrates for HSCAN
		printf("\tGetting DW CAN 01 Baudrate... ");
		int64_t baudrate = 0;
		res = icsneoc2_settings_baudrate_get(open_device, icsneoc2_netid_dwcan_01, &baudrate);
		if(res != icsneoc2_error_success) {
			print_events(description);
			icsneoc2_device_free(open_device);
			icsneoc2_enumeration_free(found_devices);
			return print_error_code("\tFailed to get baudrate", res);
		};
		printf("%" PRIu64 "mbit/s\n", baudrate);
		// Get FDbaudrates for HSCAN
		printf("\tGetting FD DW CAN 01 Baudrate... ");
		int64_t fd_baudrate = 0;
		res = icsneoc2_settings_canfd_baudrate_get(open_device, icsneoc2_netid_dwcan_01, &fd_baudrate);
		if(res != icsneoc2_error_success) {
			print_events(description);
			icsneoc2_device_free(open_device);
			icsneoc2_enumeration_free(found_devices);
			return print_error_code("\tFailed to get FD baudrate", res);
		};
		printf("%" PRIu64 "mbit/s\n", fd_baudrate);
		// Set baudrates for HSCAN
		// save_to_device: If this is set to true, the baudrate will be saved on the device
		// and will persist through a power cycle
		printf("\tSetting DW CAN 01 Baudrate... ");
		res = icsneoc2_settings_baudrate_set(open_device, icsneoc2_netid_dwcan_01, baudrate);
		if(res != icsneoc2_error_success) {
			print_events(description);
			icsneoc2_device_free(open_device);
			icsneoc2_enumeration_free(found_devices);
			return print_error_code("\tFailed to set baudrate", res);
		};
		printf("Ok\n");
		// Set FDbaudrates for HSCAN
		printf("\tSetting FD DW CAN 01 Baudrate... ");
		res = icsneoc2_settings_canfd_baudrate_set(open_device, icsneoc2_netid_dwcan_01, fd_baudrate);
		if(res != icsneoc2_error_success) {
			print_events(description);
			icsneoc2_device_free(open_device);
			icsneoc2_enumeration_free(found_devices);
			return print_error_code("\tFailed to set FD baudrate", res);
		};
		printf("Ok\n");
		// Get RTC
		printf("\tGetting RTC... ");
		res = get_and_print_rtc(open_device);
		if(res != icsneoc2_error_success) {
			print_events(description);
			icsneoc2_device_free(open_device);
			icsneoc2_enumeration_free(found_devices);
			return print_error_code("\tFailed to get RTC", res);
		}
		// Set RTC
		printf("\tSetting RTC to current time... ");
		time_t current_time = time(NULL);
		res = icsneoc2_device_rtc_set(open_device, current_time);
		if(res != icsneoc2_error_success) {
			print_events(description);
			icsneoc2_device_free(open_device);
			icsneoc2_enumeration_free(found_devices);
			return print_error_code("\tFailed to set RTC", res);
		}
		printf("Ok\n");
		// Get RTC
		printf("\tGetting RTC... ");
		res = get_and_print_rtc(open_device);
		if(res != icsneoc2_error_success) {
			print_events(description);
			icsneoc2_device_free(open_device);
			icsneoc2_enumeration_free(found_devices);
			return print_error_code("\tFailed to get RTC", res);
		}
		// Go online, start acking traffic
		printf("\tGoing online... ");
		res = icsneoc2_device_go_online(open_device, true);
		if(res != icsneoc2_error_success) {
			print_events(description);
			icsneoc2_device_free(open_device);
			icsneoc2_enumeration_free(found_devices);
			return print_error_code("\tFailed to go online", res);
		}
		// Redundant check to show how to check if the device is online, if the previous
		// icsneoc2_device_go_online call was successful we can assume we are online already
		bool is_online = false;
		res = icsneoc2_device_is_online(open_device, &is_online);
		if(res != icsneoc2_error_success) {
			print_events(description);
			icsneoc2_device_free(open_device);
			icsneoc2_enumeration_free(found_devices);
			return print_error_code("\tFailed to check if online", res);
		}
		printf("%s\n", is_online ? "Online" : "Offline");
		// Transmit CAN messages
		res = transmit_can_messages(open_device);
		if(res != icsneoc2_error_success) {
			print_events(description);
			icsneoc2_device_free(open_device);
			icsneoc2_enumeration_free(found_devices);
			return print_error_code("\tFailed to transmit CAN messages", res);
		}
		// Wait for the bus to collect some messages, requires an active bus to get messages
		printf("\tWaiting 1 second for messages...\n");
		sleep_ms(1000);
		// Get the messages
		icsneoc2_message_t* messages[20000] = {0};
		size_t message_count = 20000;
		printf("\tGetting messages from device with timeout of 3000ms on %s...\n", description);
		for(size_t i = 0; i < message_count; ++i) {
			res = icsneoc2_device_message_get(open_device, &messages[i], 0);
			if(res != icsneoc2_error_success) {
				for(size_t j = 0; j < i; ++j) {
					icsneoc2_message_free(messages[j]);
				}
				print_events(description);
				icsneoc2_device_free(open_device);
				icsneoc2_enumeration_free(found_devices);
				return print_error_code("\tFailed to get messages from device", res);
			};
			if(messages[i] == NULL) {
				// no more messages
				message_count = i;
				break;
			}
		}
		// Process the messages
		res = process_messages(messages, message_count);
		if(res != icsneoc2_error_success) {
			for(size_t i = 0; i < message_count; ++i) {
				icsneoc2_message_free(messages[i]);
			}
			print_events(description);
			icsneoc2_device_free(open_device);
			icsneoc2_enumeration_free(found_devices);
			return print_error_code("\tFailed to process messages", res);
		}
		for(size_t i = 0; i < message_count; ++i) {
			icsneoc2_message_free(messages[i]);
		}
		// Finally, close the device.
		printf("\tClosing device: %s...\n", description);
		res = icsneoc2_device_close(open_device);
		if(res != icsneoc2_error_success) {
			print_events(description);
			icsneoc2_device_free(open_device);
			icsneoc2_enumeration_free(found_devices);
			return print_error_code("\tFailed to close device", res);
		};
		// Print device events
		print_events(description);
		icsneoc2_device_free(open_device);
	}
	icsneoc2_enumeration_free(found_devices);
	printf("\n");
	return 0;
}

icsneoc2_error_t get_and_print_rtc(icsneoc2_device_t* device) {
	time_t unix_epoch = 0;
	icsneoc2_error_t res = icsneoc2_device_rtc_get(device, &unix_epoch);
	if(res != icsneoc2_error_success) {
		return res;
	}
	char rtc_time[32] = {0};
	strftime(rtc_time, sizeof(rtc_time), "%Y-%m-%d %H:%M:%S", localtime(&unix_epoch));
	printf("RTC: %lld %s\n", (long long)unix_epoch, rtc_time);

	return icsneoc2_error_success;
}

void print_events(const char* device_description) {
	icsneoc2_event_t* events[1024] = {0};
	size_t events_count = 1024;
	for(size_t i = 0; i < events_count; ++i) {
		// no device filter, get all events
		icsneoc2_error_t res = icsneoc2_event_get(&events[i], NULL);
		if(res != icsneoc2_error_success) {
			for(size_t j = 0; j < i; ++j) {
				icsneoc2_event_free(events[j]);
			}
			(void)print_error_code("\tFailed to get device events", res);
			return;
		}
		if(events[i] == NULL) {
			events_count = i;
			break;
		}
	}
	// Loop over each event and describe it.
	for(size_t i = 0; i < events_count; i++) {
		char event_description[255] = {0};
		size_t event_description_length = 255;
		icsneoc2_error_t res = icsneoc2_event_description_get(events[i], event_description, &event_description_length);
		if(res != icsneoc2_error_success) {
			print_error_code("\tFailed to get event description", res);
			continue;
		}
		printf("\t%s: Event %zu: %s\n", device_description, i, event_description);
	}
	for(size_t i = 0; i < events_count; i++) {
		icsneoc2_event_free(events[i]);
	}
	printf("\t%s: Received %zu events\n", device_description, events_count);
}

int process_messages(icsneoc2_message_t** messages, size_t messages_count) {
	// Print the type and bus type of each message
	size_t tx_count = 0;
	size_t can_error_count = 0;
	for(size_t i = 0; i < messages_count; i++) {
		icsneoc2_message_t* message = messages[i];

		// Check for CAN error messages
		bool is_can_error = false;
		icsneoc2_error_t res = icsneoc2_message_is_can_error(message, &is_can_error);
		if(res != icsneoc2_error_success) {
			return print_error_code("\tFailed to check if message is a CAN error", res);
		}
		if(is_can_error) {
			uint8_t tec = 0;
			uint8_t rec = 0;
			icsneoc2_can_error_code_t error_code = 0;
			icsneoc2_can_error_code_t data_error_code = 0;
			icsneoc2_message_can_error_flags_t error_flags = 0;
			icsneoc2_netid_t netid = 0;
			res = icsneoc2_message_netid_get(message, &netid);
			res += icsneoc2_message_can_error_props_get(message, &tec, &rec, &error_code, &data_error_code, &error_flags);
			if(res != icsneoc2_error_success) {
				return print_error_code("\tFailed to get CAN error properties", res);
			}
			char netid_name[128] = {0};
			size_t netid_name_length = 128;
			icsneoc2_netid_name_get(netid, netid_name, &netid_name_length);
			printf("\t%zd) CAN Error on %s (0x%x): TEC=%u REC=%u ErrorCode=%u DataErrorCode=%u%s%s%s\n",
				i, netid_name, netid, tec, rec, error_code, data_error_code,
				(error_flags & ICSNEOC2_MESSAGE_CAN_ERROR_FLAGS_BUS_OFF) ? " [BusOff]" : "",
				(error_flags & ICSNEOC2_MESSAGE_CAN_ERROR_FLAGS_ERROR_PASSIVE) ? " [ErrorPassive]" : "",
				(error_flags & ICSNEOC2_MESSAGE_CAN_ERROR_FLAGS_ERROR_WARN) ? " [ErrorWarn]" : "");
			can_error_count++;
			continue;
		}

		bool is_frame = false;
		res = icsneoc2_message_is_frame(message, &is_frame);
		if(res != icsneoc2_error_success) {
			return print_error_code("\tFailed to check if message is a frame", res);
		}
		if(!is_frame) {
			printf("Ignoring non-frame message at index %zu\n", i);
			continue;
		}
		icsneoc2_network_type_t network_type;
		res = icsneoc2_message_network_type_get(message, &network_type);
		if(res != icsneoc2_error_success) {
			return print_error_code("\tFailed to get message network type", res);
		}

		char network_type_name[128] = {0};
		size_t network_type_name_length = 128;
		res = icsneoc2_network_type_name_get(network_type, network_type_name, &network_type_name_length);
		if(res != icsneoc2_error_success) {
			return print_error_code("\tFailed to get network type name", res);
		}
		bool is_tx = false;
		res = icsneoc2_message_is_transmit(message, &is_tx);
		if(res != icsneoc2_error_success) {
			return print_error_code("\tFailed to get message is transmit", res);
		}
		if(is_tx) {
			tx_count++;
			continue;
		}

		printf("\t%zd) network type: %s (%u)\n", i, network_type_name, network_type);

		if(network_type == icsneoc2_network_type_can) {
			uint64_t arbid = 0;
			int32_t dlc = 0;
			icsneoc2_netid_t netid = 0;
			icsneoc2_message_can_flags_t can_flags = 0;
			uint8_t data[64] = {0};
			size_t data_length = 64;
			char netid_name[128] = {0};
			size_t netid_name_length = 128;
			icsneoc2_error_t result = icsneoc2_message_netid_get(message, &netid);
			result += icsneoc2_netid_name_get(netid, netid_name, &netid_name_length);
			result += icsneoc2_message_can_props_get(message, &arbid, &can_flags);
			result += icsneoc2_message_data_get(message, data, &data_length);
			if(result != icsneoc2_error_success) {
				printf("\tFailed get get CAN parameters (error: %u) for index %zu\n", result, i);
				continue;
			}
			bool is_remote = (can_flags & ICSNEOC2_MESSAGE_CAN_FLAGS_RTR) != 0;
			bool is_extended = (can_flags & ICSNEOC2_MESSAGE_CAN_FLAGS_IDE) != 0;
			bool is_canfd = (can_flags & ICSNEOC2_MESSAGE_CAN_FLAGS_FDF) != 0;
			bool is_brs = (can_flags & ICSNEOC2_MESSAGE_CAN_FLAGS_BRS) != 0;
			bool is_esi = (can_flags & ICSNEOC2_MESSAGE_CAN_FLAGS_ESI) != 0;
			bool tx_aborted = (can_flags & ICSNEOC2_MESSAGE_CAN_FLAGS_TX_ABORTED) != 0;
			bool tx_lost_arb = (can_flags & ICSNEOC2_MESSAGE_CAN_FLAGS_TX_LOST_ARB) != 0;
			bool tx_error = (can_flags & ICSNEOC2_MESSAGE_CAN_FLAGS_TX_ERROR) != 0;
			dlc = (int32_t)data_length;

			printf("\t  NetID: %s (0x%x)\tArbID: 0x%llx\tDLC: %u\tLen: %zu\n", netid_name, netid, (unsigned long long)arbid, dlc, data_length);
			printf("\t  Flags:%s%s%s%s%s%s%s%s\n",
				is_remote ? " RTR" : "",
				is_extended ? " IDE" : "",
				is_canfd ? " FDF" : "",
				is_brs ? " BRS" : "",
				is_esi ? " ESI" : "",
				tx_aborted ? " TX_ABORTED" : "",
				tx_lost_arb ? " TX_LOST_ARB" : "",
				tx_error ? " TX_ERROR" : "");
			printf("\t  Data: [");
			for(size_t x = 0; x < data_length; x++) {
				printf(" 0x%x", data[x]);
			}
			printf(" ]\n");
		}
	}
	printf("\tReceived %zu messages total, %zu were TX messages, %zu were CAN errors\n", messages_count, tx_count, can_error_count);

	return icsneoc2_error_success;
}

int transmit_can_messages(icsneoc2_device_t* device) {
	uint64_t counter = 0;
	const size_t msg_count = 100;
	printf("\tTransmitting %zd messages...\n", msg_count);
	for(size_t i = 0; i < msg_count; i++) {
		// Create the message
		icsneoc2_message_t* message = NULL;
		icsneoc2_error_t res = icsneoc2_message_can_create(&message);
		if(res != icsneoc2_error_success) {
			return print_error_code("\tFailed to create messages", res);
		}
		// Set the message attributes
		res = icsneoc2_message_netid_set(message, icsneoc2_netid_dwcan_01);
		uint64_t arb_id = 0x10;
		uint64_t flags = ICSNEOC2_MESSAGE_CAN_FLAGS_BRS | ICSNEOC2_MESSAGE_CAN_FLAGS_IDE | ICSNEOC2_MESSAGE_CAN_FLAGS_FDF;
		res += icsneoc2_message_can_props_set(message, &arb_id, &flags);
		res += icsneoc2_message_data_set(message, (uint8_t*)&counter, sizeof(counter));
		if(res != icsneoc2_error_success) {
			icsneoc2_message_free(message);
			return print_error_code("\tFailed to modify message", res);
		}
		res = icsneoc2_device_message_transmit(device, message);
		res += icsneoc2_message_free(message);
		if(res != icsneoc2_error_success) {
			return print_error_code("\tFailed to transmit message", res);
		}
		counter++;
	}

	return icsneoc2_error_success;
}

Disk Format

Download example

#include <icsneo/icsneoc2.h>

#include <stdio.h>
#include <inttypes.h>

/**
 * Prints an error message with the given string and error code.
 *
 * @param message The message to print.
 * @param error The error code to print.
 * @return error as int
 */
static int print_error_code(const char* message, icsneoc2_error_t error) {
	char error_str[64];
	size_t error_str_len = sizeof(error_str);
	icsneoc2_error_code_get(error, error_str, &error_str_len);
	printf("%s: \"%s\" (%u)\n", message, error_str, error);
	return (int)error;
}

/**
 * Progress callback invoked periodically during disk formatting.
 *
 * @param sectors_formatted Number of sectors formatted so far.
 * @param total_sectors Total number of sectors to format.
 * @param user_data Unused opaque pointer.
 * @return icsneoc2_disk_format_directive_continue to keep formatting.
 */
static icsneoc2_disk_format_directive_t format_progress(uint64_t sectors_formatted, uint64_t total_sectors, void* user_data) {
	(void)user_data;
	double pct = total_sectors > 0 ? (100.0 * (double)sectors_formatted / (double)total_sectors) : 0.0;
	printf("\r  Progress: %" PRIu64 " / %" PRIu64 " sectors  (%d%%)", sectors_formatted, total_sectors, (int)pct);
	fflush(stdout);
	return icsneoc2_disk_format_directive_continue;
}

int main() {
	/* Open the first available device (no online needed for formatting) */
	printf("Opening first available device...\n");
	icsneoc2_device_t* device = NULL;
	icsneoc2_error_t res = icsneoc2_device_open_first(0, ICSNEOC2_OPEN_OPTIONS_NONE, &device);
	if(res != icsneoc2_error_success) {
		return print_error_code("\tFailed to open first device", res);
	}

	/* Get a description of the opened device */
	char description[255] = {0};
	size_t description_length = sizeof(description);
	res = icsneoc2_device_description_get(device, description, &description_length);
	if(res != icsneoc2_error_success) {
		icsneoc2_device_free(device);
		return print_error_code("\tFailed to get device description", res);
	}
	printf("\tOpened device: %s\n", description);

	/* Check disk formatting support */
	bool supported = false;
	res = icsneoc2_device_supports_disk_formatting(device, &supported);
	if(res != icsneoc2_error_success) {
		print_error_code("\tFailed to check disk formatting support", res);
		icsneoc2_device_close(device);
		icsneoc2_device_free(device);
		return -1;
	}
	if(!supported) {
		printf("\terror: %s does not support disk formatting\n", description);
		icsneoc2_device_close(device);
		icsneoc2_device_free(device);
		return -1;
	}

	size_t disk_count = 0;
	icsneoc2_device_disk_count_get(device, &disk_count);
	printf("\tDisk count: %zu\n", disk_count);

	/* Query disk details */
	printf("\tQuerying disk details... ");
	fflush(stdout);
	icsneoc2_disk_details_t* details = NULL;
	res = icsneoc2_device_disk_details_get(device, &details);
	if(res != icsneoc2_error_success) {
		printf("FAIL\n");
		print_error_code("\tFailed to get disk details", res);
		icsneoc2_device_close(device);
		icsneoc2_device_free(device);
		return -1;
	}
	printf("OK\n");

	/* Display current state */
	icsneoc2_disk_layout_t layout = 0;
	icsneoc2_disk_details_layout_get(details, &layout);
	printf("\t  Layout        : %s\n", layout == icsneoc2_disk_layout_raid0 ? "RAID0" : "Spanned");

	size_t detail_count = 0;
	icsneoc2_disk_details_count_get(details, &detail_count);
	for(size_t i = 0; i < detail_count; i++) {
		icsneoc2_disk_format_flags_t flags = 0;
		icsneoc2_disk_details_flags_get(details, i, &flags);
		printf("\t  Disk [%zu]:\n", i);
		printf("\t    Present     : %s\n", (flags & ICSNEOC2_DISK_FORMAT_FLAGS_PRESENT) ? "yes" : "no");
		printf("\t    Initialized : %s\n", (flags & ICSNEOC2_DISK_FORMAT_FLAGS_INITIALIZED) ? "yes" : "no");
		printf("\t    Formatted   : %s\n", (flags & ICSNEOC2_DISK_FORMAT_FLAGS_FORMATTED) ? "yes" : "no");
		if(flags & ICSNEOC2_DISK_FORMAT_FLAGS_PRESENT) {
			uint64_t sectors = 0, bps = 0;
			icsneoc2_disk_details_size_get(details, i, &sectors, &bps);
			printf("\t    Size        : %" PRIu64 " MB (%" PRIu64 " sectors x %" PRIu64 " bytes)\n",
				(sectors * bps) / (1024 * 1024), sectors, bps);
		}
	}

	/* Build format config: mark present disks for formatting */
	bool any_present = false;
	for(size_t i = 0; i < detail_count; i++) {
		icsneoc2_disk_format_flags_t flags = 0;
		icsneoc2_disk_details_flags_get(details, i, &flags);
		if(flags & ICSNEOC2_DISK_FORMAT_FLAGS_PRESENT) {
			flags |= ICSNEOC2_DISK_FORMAT_FLAGS_FORMATTED;
			icsneoc2_disk_details_flags_set(details, i, flags);
			any_present = true;
		}
	}
	icsneoc2_disk_details_full_format_set(details, false); /* Quick format */

	if(!any_present) {
		printf("\n\terror: no disks are present in the device\n");
		icsneoc2_disk_details_free(details);
		icsneoc2_device_close(device);
		icsneoc2_device_free(device);
		return -1;
	}

	/* Confirm */
	printf("\n\tThis will format the disk(s) in %s.\n", description);
	printf("\tAll existing data will be lost. Continue? [y/N]: ");
	char confirm[8] = {0};
	if(scanf("%7s", confirm) != 1 || (confirm[0] != 'y' && confirm[0] != 'Y')) {
		printf("\tAborted.\n");
		icsneoc2_disk_details_free(details);
		icsneoc2_device_close(device);
		icsneoc2_device_free(device);
		return 0;
	}

	/* Format */
	printf("\n\tStarting format...\n");
	res = icsneoc2_device_format_disk(device, details, format_progress, NULL);
	printf("\n"); /* newline after progress line */

	if(res != icsneoc2_error_success) {
		print_error_code("\tFormat failed", res);
		icsneoc2_disk_details_free(details);
		icsneoc2_device_close(device);
		icsneoc2_device_free(device);
		return -1;
	}
	printf("\tFormat complete!\n");
	icsneoc2_disk_details_free(details);

	/* Verify */
	printf("\n\tVerifying disk state after format... ");
	fflush(stdout);
	icsneoc2_disk_details_t* post_details = NULL;
	res = icsneoc2_device_disk_details_get(device, &post_details);
	if(res != icsneoc2_error_success) {
		printf("FAIL (could not re-query disk details)\n");
	} else {
		printf("OK\n");
		size_t post_count = 0;
		icsneoc2_disk_details_count_get(post_details, &post_count);
		for(size_t i = 0; i < post_count; i++) {
			icsneoc2_disk_format_flags_t flags = 0;
			icsneoc2_disk_details_flags_get(post_details, i, &flags);
			printf("\t  Disk [%zu] formatted: %s\n", i, (flags & ICSNEOC2_DISK_FORMAT_FLAGS_FORMATTED) ? "yes" : "no");
		}
		icsneoc2_disk_details_free(post_details);
	}

	printf("\tClosing device: %s...\n", description);
	icsneoc2_device_close(device);
	icsneoc2_device_free(device);
	return 0;
}

Read Messages

Download example

#include <icsneo/icsneoc2.h>
#include <icsneo/icsneoc2settings.h>
#include <icsneo/icsneoc2messages.h>

#include <stdio.h>
#include <inttypes.h>
#include <time.h>

#ifdef _WIN32
#include <windows.h>
#else
#include <unistd.h>
#endif


/**
 * Sleeps for a specified number of milliseconds using Sleep() on Windows and sleep() on *nix.
 *
 * @param ms The number of milliseconds to sleep.
 */
void sleep_ms(uint32_t ms) {
#ifdef _WIN32
	Sleep(ms);
#else
	sleep(ms / 1000);
#endif
}

/**
 * Prints all events
 *
 * @param device_description A description of the device used in the output.
 */
void print_events(const char* device_description);

/**
 * Prints an error message with the given string and error code.
 *
 * If the error code is not icsneoc2_error_success, prints the error string for the given error code
 * and returns the error code.
 *
 * @param message The message to print.
 * @param error The error code to print.
 * @return error as int
 */
int print_error_code(const char* message, icsneoc2_error_t error) {
	char error_str[64];
	size_t error_str_len = sizeof(error_str);
	icsneoc2_error_t res = icsneoc2_error_code_get(error, error_str, &error_str_len);
	if(res != icsneoc2_error_success) {
		printf("%s: Failed to get string for error code %d with error code %d\n", message, error, res);
		return res;
	}
	printf("%s: \"%s\" (%u)\n", message, error_str, error);
	return (int)error;
}

/**
 * Processes a list of messages from a device.
 *
 * This function iterates over a given array of messages received from a specified device.
 * For each message in the array, it retrieves and prints the message type and bus type.
 * If an error occurs while retrieving these details, an error message is printed.
 *
 * @param messages An array of pointers to icsneoc2_message_t structures containing the messages to process.
 * @param messages_count The number of messages in the messages array.
 *
 * @return An icsneoc2_error_t value indicating success or failure of the message processing.
 */
int process_message(icsneoc2_message_t** messages, size_t messages_count);

int transmit_can_messages(icsneoc2_device_t* device);

int main() {
	// Open the first available device with default options
	printf("Opening first available device...\n");
	icsneoc2_device_t* open_device = NULL;
	icsneoc2_error_t res = icsneoc2_device_open_first(0, icsneoc2_open_options_default, &open_device);
	if(res != icsneoc2_error_success) {
		return print_error_code("\tFailed to open first device", res);
	};

	// Get a description of the opened device
	char description[255] = {0};
	size_t description_length = 255;
	res = icsneoc2_device_description_get(open_device, description, &description_length);
	if(res != icsneoc2_error_success) {
		icsneoc2_device_free(open_device);
		return print_error_code("\tFailed to get device description", res);
	};
	printf("\tOpened device: %s\n", description);

	// Transmit messages for debugging purposes
	// transmit_can_messages(open_device);
	// sleep_ms(1000);

	// Get the messages
	icsneoc2_message_t* messages[20000] = {0};
	size_t message_count = 20000;
	time_t start_time = time(NULL);
	printf("\tGetting messages from device with timeout of 3000ms on %s...\n", description);
	for(size_t i = 0; i < message_count; ++i) {
		res = icsneoc2_device_message_get(open_device, &messages[i], 0);
		if(res != icsneoc2_error_success) {
			print_events(description);
			icsneoc2_device_free(open_device);
			return print_error_code("\tFailed to get messages from device", res);
		};
		if(messages[i] == NULL) {
			// no more messages
			message_count = i;
			break;
		}
	}
	if(res != icsneoc2_error_success) {
		print_events(description);
		icsneoc2_device_free(open_device);
		return print_error_code("\tFailed to get messages from device", res);
	}
	time_t end_time = time(NULL);
	printf("\tGot %zu messages in %lld seconds\n", message_count, (long long)(end_time - start_time));
	// Process the messages
	res = process_message(messages, message_count);
	if(res != icsneoc2_error_success) {
		print_events(description);
		icsneoc2_device_free(open_device);
		return print_error_code("\tFailed to process messages", res);
	}
	// Finally, close the device.
	printf("\tClosing device: %s...\n", description);
	res = icsneoc2_device_close(open_device);
	if(res != icsneoc2_error_success) {
		print_events(description);
		icsneoc2_device_free(open_device);
		return print_error_code("\tFailed to close device", res);
	};
	icsneoc2_device_free(open_device);

	printf("\n");
	return 0;
}

void print_events(const char* device_description) {
	icsneoc2_event_t* events[1024] = {0};
	size_t events_count = 1024;
	for(size_t i = 0; i < events_count; ++i) {
		// no device filter, get all events
		icsneoc2_error_t res = icsneoc2_event_get(&events[i], NULL);
		if(res != icsneoc2_error_success) {
			(void)print_error_code("\tFailed to get device events", res);
			return;
		}
		if(events[i] == NULL) {
			events_count = i;
			break;
		}
	}
	// Loop over each event and describe it.
	for(size_t i = 0; i < events_count; i++) {
		char event_description[255] = {0};
		size_t event_description_length = 255;
		icsneoc2_error_t res = icsneoc2_event_description_get(events[i], event_description, &event_description_length);
		if(res != icsneoc2_error_success) {
			print_error_code("\tFailed to get event description", res);
			continue;
		}
		printf("\t%s: Event %zu: %s\n", device_description, i, event_description);
	}
	for(size_t i = 0; i < events_count; i++) {
		icsneoc2_event_free(events[i]);
	}
	printf("\t%s: Received %zu events\n", device_description, events_count);
}

int process_message(icsneoc2_message_t** messages, size_t messages_count) {
	// Print the type and bus type of each message
	size_t tx_count = 0;
	size_t can_error_count = 0;
	icsneoc2_error_t res = icsneoc2_error_success;
	for(size_t i = 0; i < messages_count; i++) {
		icsneoc2_message_t* message = messages[i];

		bool is_can_error = false;
		res = icsneoc2_message_is_can_error(message, &is_can_error);
		if(res != icsneoc2_error_success) {
			return print_error_code("\tFailed to check if message is a CAN error", res);
		}
		if(is_can_error) {
			icsneoc2_network_type_t network_type;
			uint8_t tec = 0;
			uint8_t rec = 0;
			icsneoc2_can_error_code_t error_code = 0;
			icsneoc2_can_error_code_t data_error_code = 0;
			icsneoc2_message_can_error_flags_t error_flags = 0;
			icsneoc2_netid_t netid = 0;
			char network_type_name[128] = {0};
			size_t network_type_name_length = 128;
			char netid_name[128] = {0};
			size_t netid_name_length = 128;

			res = icsneoc2_message_network_type_get(message, &network_type);
			res += icsneoc2_network_type_name_get(network_type, network_type_name, &network_type_name_length);
			res += icsneoc2_message_netid_get(message, &netid);
			res += icsneoc2_netid_name_get(netid, netid_name, &netid_name_length);
			res += icsneoc2_message_can_error_props_get(message, &tec, &rec, &error_code, &data_error_code, &error_flags);
			if(res != icsneoc2_error_success) {
				return print_error_code("\tFailed to get CAN error properties", res);
			}

			printf("\t%zd) CAN Error on %s [%s] (0x%x): TEC=%u REC=%u ErrorCode=%u DataErrorCode=%u%s%s%s\n",
				i, netid_name, network_type_name, netid, tec, rec, error_code, data_error_code,
				(error_flags & ICSNEOC2_MESSAGE_CAN_ERROR_FLAGS_BUS_OFF) ? " [BusOff]" : "",
				(error_flags & ICSNEOC2_MESSAGE_CAN_ERROR_FLAGS_ERROR_PASSIVE) ? " [ErrorPassive]" : "",
				(error_flags & ICSNEOC2_MESSAGE_CAN_ERROR_FLAGS_ERROR_WARN) ? " [ErrorWarn]" : "");
			can_error_count++;
			continue;
		}

		bool is_frame = false;
		res = icsneoc2_message_is_frame(message, &is_frame);
		if(res != icsneoc2_error_success) {
			return print_error_code("\tFailed to check if message is a frame", res);
		}
		if(!is_frame) {
			printf("Ignoring non-frame message at index %zu\n", i);
			continue;
		}
		icsneoc2_network_type_t network_type;
		res = icsneoc2_message_network_type_get(message, &network_type);
		if(res != icsneoc2_error_success) {
			return print_error_code("\tFailed to get message network type", res);
		}

		char network_type_name[128] = {0};
		size_t network_type_name_length = 128;
		res = icsneoc2_network_type_name_get(network_type, network_type_name, &network_type_name_length);
		if(res != icsneoc2_error_success) {
			return print_error_code("\tFailed to get message bus type name", res);
		}
		
		if(res != icsneoc2_error_success) {
			return print_error_code("\tFailed to get message is transmit", res);
		}

		printf("\t%zd) network type: %s (%u)\n", i, network_type_name, network_type);
		if(network_type == icsneoc2_network_type_can) {
			uint64_t arbid = 0;
			int32_t dlc = 0;
			icsneoc2_netid_t netid = 0;
			icsneoc2_message_can_flags_t can_flags = 0;
			uint8_t data[64] = {0};
			size_t data_length = 64;
			char netid_name[128] = {0};
			size_t netid_name_length = 128;
			bool is_error = false;
			bool is_tx = false;
			icsneoc2_error_t result = icsneoc2_message_netid_get(message, &netid);
			result += icsneoc2_netid_name_get(netid, netid_name, &netid_name_length);
			result += icsneoc2_message_can_props_get(message, &arbid, &can_flags);
			result += icsneoc2_message_data_get(message, data, &data_length);
			result += icsneoc2_message_is_transmit(message, &is_tx);
			result += icsneoc2_message_is_error(message, &is_error);
			if(result != icsneoc2_error_success) {
				printf("\tFailed get get CAN parameters (error: %u) for index %zu\n", result, i);
				continue;
			}
			tx_count += is_tx ? 1 : 0;
			bool is_remote = (can_flags & ICSNEOC2_MESSAGE_CAN_FLAGS_RTR) != 0;
			bool is_extended = (can_flags & ICSNEOC2_MESSAGE_CAN_FLAGS_IDE) != 0;
			bool is_canfd = (can_flags & ICSNEOC2_MESSAGE_CAN_FLAGS_FDF) != 0;
			bool is_brs = (can_flags & ICSNEOC2_MESSAGE_CAN_FLAGS_BRS) != 0;
			bool is_esi = (can_flags & ICSNEOC2_MESSAGE_CAN_FLAGS_ESI) != 0;
			bool tx_aborted = (can_flags & ICSNEOC2_MESSAGE_CAN_FLAGS_TX_ABORTED) != 0;
			bool tx_lost_arb = (can_flags & ICSNEOC2_MESSAGE_CAN_FLAGS_TX_LOST_ARB) != 0;
			bool tx_error = (can_flags & ICSNEOC2_MESSAGE_CAN_FLAGS_TX_ERROR) != 0;
			dlc = (int32_t)data_length;

			printf("\t  %s%s\n", is_tx ? "TX" : "RX", is_error ? " [Error]" : "");
			printf("\t  NetID: %s (0x%x)\tArbID: 0x%llx\tDLC: %u\tLen: %zu\n", netid_name, netid, (unsigned long long)arbid, dlc, data_length);
			printf("\t  Flags:%s%s%s%s%s%s%s%s\n",
				is_remote ? " RTR" : "",
				is_extended ? " IDE" : "",
				is_canfd ? " FDF" : "",
				is_brs ? " BRS" : "",
				is_esi ? " ESI" : "",
				tx_aborted ? " TX_ABORTED" : "",
				tx_lost_arb ? " TX_LOST_ARB" : "",
				tx_error ? " TX_ERROR" : "");
			printf("\t  Data: [");
			for(size_t x = 0; x < data_length; x++) {
				printf(" 0x%x", data[x]);
			}
			printf(" ]\n");
		}
	}
	printf("\tReceived %zu messages total, %zu were TX messages, %zu were CAN errors\n", messages_count, tx_count, can_error_count);

	return icsneoc2_error_success;
}

int transmit_can_messages(icsneoc2_device_t* device) {
	uint64_t counter = 0;
	const size_t msg_count = 10;
	printf("\tTransmitting %zd messages...\n", msg_count);
	for(size_t i = 0; i < msg_count; i++) {
		// Create the message
		icsneoc2_message_t* message = NULL;
		icsneoc2_error_t res = icsneoc2_message_can_create(&message);
		if(res != icsneoc2_error_success) {
			return print_error_code("\tFailed to create messages", res);
		}
		// Set the message attributes
		res = icsneoc2_message_netid_set(message, icsneoc2_netid_dwcan_01);
		uint64_t arb_id = 0x10;
		uint64_t flags = 0;
		res += icsneoc2_message_can_props_set(message, &arb_id, &flags);
		res += icsneoc2_message_data_set(message, (uint8_t*)&counter, sizeof(counter));
		if(res != icsneoc2_error_success) {
			icsneoc2_message_free(message);
			return print_error_code("\tFailed to modify message", res);
		}
		res = icsneoc2_device_message_transmit(device, message);
		res += icsneoc2_message_free(message);
		if(res != icsneoc2_error_success) {
			return print_error_code("\tFailed to transmit message", res);
		}
		counter++;
	}

	return icsneoc2_error_success;
}

Device Info

Download example

#include <icsneo/icsneoc2.h>

#include <stdio.h>
#include <inttypes.h>

int print_error_code(const char* message, icsneoc2_error_t error) {
	char error_str[64];
	size_t error_str_len = sizeof(error_str);
	icsneoc2_error_t res = icsneoc2_error_code_get(error, error_str, &error_str_len);
	if(res != icsneoc2_error_success) {
		printf("%s: Failed to get string for error code %d with error code %d\n", message, error, res);
		return res;
	}
	printf("%s: \"%s\" (%u)\n", message, error_str, error);
	return (int)error;
}

int main() {
	icsneoc2_error_t res;

	/* ===== Device Selection ===== */
	printf("Searching for devices...\n");
	icsneoc2_device_info_t* found_devices = NULL;
	res = icsneoc2_device_enumerate(0, &found_devices);
	if(res != icsneoc2_error_success) {
		return print_error_code("Failed to enumerate devices", res);
	}
	if(!found_devices) {
		printf("No devices found.\n");
		return 1;
	}

	/* Count and display devices */
	int device_count = 0;
	for(icsneoc2_device_info_t* cur = found_devices; cur; cur = icsneoc2_device_info_next(cur)) {
		char desc[128] = {0};
		size_t desc_len = sizeof(desc);
		icsneoc2_device_info_description_get(cur, desc, &desc_len);

		char serial[32] = {0};
		size_t serial_len = sizeof(serial);
		icsneoc2_device_info_serial_get(cur, serial, &serial_len);

		printf("  [%d] %s (Serial: %s)\n", device_count + 1, desc, serial);
		device_count++;
	}

	int device_choice;
	printf("Select device (1-%d): ", device_count);
	if(scanf("%d", &device_choice) != 1 || device_choice < 1 || device_choice > device_count) {
		printf("Invalid selection.\n");
		icsneoc2_enumeration_free(found_devices);
		return 1;
	}

	/* Find the selected device_info node */
	icsneoc2_device_info_t* selected_info = found_devices;
	for(int i = 1; i < device_choice; i++) {
		selected_info = icsneoc2_device_info_next(selected_info);
	}

	/* Open the selected device */
	icsneoc2_device_t* device = NULL;
	res = icsneoc2_device_create(selected_info, &device);
	if(res != icsneoc2_error_success) {
		icsneoc2_enumeration_free(found_devices);
		return print_error_code("Failed to create device from device info", res);
	}
	res = icsneoc2_device_open(device, icsneoc2_open_options_default);
	icsneoc2_enumeration_free(found_devices);
	if(res != icsneoc2_error_success) {
		icsneoc2_device_free(device);
		return print_error_code("Failed to open device", res);
	}

	char description[128] = {0};
	size_t description_length = sizeof(description);
	icsneoc2_device_description_get(device, description, &description_length);
	printf("\nOpened device: %s\n\n", description);

	/* ===== Serial Number ===== */
	char serial[32] = {0};
	size_t serial_len = sizeof(serial);
	res = icsneoc2_device_serial_get(device, serial, &serial_len);
	if(res == icsneoc2_error_success) {
		printf("Serial:     %s\n", serial);
	} else {
		print_error_code("Failed to get serial", res);
	}

	/* ===== PCB Serial Number ===== */
	uint8_t pcbsn[16] = {0};
	size_t pcbsn_len = sizeof(pcbsn);
	res = icsneoc2_device_pcb_serial_get(device, pcbsn, &pcbsn_len);
	if(res == icsneoc2_error_success) {
		printf("PCB Serial: ");
		for(size_t i = 0; i < pcbsn_len; i++) {
			printf("%c", pcbsn[i]);
		}
		printf("\n");
	} else {
		print_error_code("Failed to get PCB serial (device may not support it)", res);
	}

	/* ===== MAC Address ===== */
	uint8_t mac[6] = {0};
	size_t mac_len = sizeof(mac);
	res = icsneoc2_device_mac_address_get(device, mac, &mac_len);
	if(res == icsneoc2_error_success) {
		printf("MAC:        ");
		for(size_t i = 0; i < mac_len; i++) {
			if(i > 0) printf(":");
			printf("%02X", mac[i]);
		}
		printf("\n");
	} else {
		print_error_code("Failed to get MAC address (device may not support it)", res);
	}

	/* Cleanup */
	printf("\nClosing device... ");
	res = icsneoc2_device_close(device);
	printf("%s\n", res == icsneoc2_error_success ? "OK" : "FAIL");

	icsneoc2_device_free(device);
	return 0;
}

LIN

Download example

/* Note: This example requires LIN 1 and LIN 2 channels to be connected on the device */

#include <icsneo/icsneoc2.h>
#include <icsneo/icsneoc2messages.h>
#include <icsneo/icsneoc2settings.h>

#ifdef _WIN32
#include <windows.h>
#else
#include <unistd.h>
#endif

#include <stdio.h>
#include <inttypes.h>

void sleep_ms(uint32_t ms) {
#ifdef _WIN32
	Sleep(ms);
#else
	sleep(ms / 1000);
#endif
}

int print_error_code(const char* message, icsneoc2_error_t error) {
	char error_str[64];
	size_t error_str_len = sizeof(error_str);
	icsneoc2_error_t res = icsneoc2_error_code_get(error, error_str, &error_str_len);
	if(res != icsneoc2_error_success) {
		printf("%s: Failed to get string for error code %d with error code %d\n", message, error, res);
		return res;
	}
	printf("%s: \"%s\" (%u)\n", message, error_str, error);
	return (int)error;
}

void print_events(const char* device_description) {
	icsneoc2_event_t* events[1024] = {0};
	size_t events_count = 1024;
	for(size_t i = 0; i < events_count; ++i) {
		icsneoc2_error_t res = icsneoc2_event_get(&events[i], NULL);
		if(res != icsneoc2_error_success) {
			for(size_t j = 0; j < i; ++j) {
				icsneoc2_event_free(events[j]);
			}
			(void)print_error_code("\tFailed to get device events", res);
			return;
		}
		if(events[i] == NULL) {
			events_count = i;
			break;
		}
	}
	for(size_t i = 0; i < events_count; i++) {
		char event_description[255] = {0};
		size_t event_description_length = 255;
		icsneoc2_error_t res = icsneoc2_event_description_get(events[i], event_description, &event_description_length);
		if(res != icsneoc2_error_success) {
			print_error_code("\tFailed to get event description", res);
			continue;
		}
		printf("\t%s: Event %zu: %s\n", device_description, i, event_description);
	}
	for(size_t i = 0; i < events_count; i++) {
		icsneoc2_event_free(events[i]);
	}
	printf("\t%s: Received %zu events\n", device_description, events_count);
}

void process_lin_messages(icsneoc2_message_t** messages, size_t count) {
	for(size_t i = 0; i < count; i++) {
		bool is_lin = false;
		icsneoc2_error_t res = icsneoc2_message_is_lin(messages[i], &is_lin);
		if(res != icsneoc2_error_success || !is_lin)
			continue;

		uint8_t id = 0;
		uint8_t protected_id = 0;
		uint8_t checksum = 0;
		icsneoc2_lin_msg_type_t msg_type = 0;
		bool is_enhanced_checksum = false;
		res = icsneoc2_message_lin_props_get(messages[i], &id, &protected_id, &checksum, &msg_type, &is_enhanced_checksum);
		if(res != icsneoc2_error_success) {
			print_error_code("\tFailed to get LIN properties", res);
			continue;
		}

		icsneoc2_netid_t netid = 0;
		icsneoc2_message_netid_get(messages[i], &netid);
		char netid_name[128] = {0};
		size_t netid_name_length = 128;
		icsneoc2_netid_name_get(netid, netid_name, &netid_name_length);

		uint8_t data[64] = {0};
		size_t data_length = 64;
		icsneoc2_message_data_get(messages[i], data, &data_length);

		icsneoc2_lin_err_flags_t err_flags = 0;
		icsneoc2_message_lin_err_flags_get(messages[i], &err_flags);

		printf("\t%s RX | ID: 0x%02x | Protected ID: 0x%02x\n", netid_name, id, protected_id);
		printf("\tData: [");
		for(size_t x = 0; x < data_length; x++) {
			printf(" 0x%02x", data[x]);
		}
		printf(" ]\n");
		printf("\tChecksum type: %s\n", is_enhanced_checksum ? "Enhanced" : "Classic");
		printf("\tChecksum: 0x%02x\n", checksum);
		printf("\tChecksum valid: %s\n\n", (!(err_flags & ICSNEOC2_LIN_ERR_CHECKSUM_MATCH)) ? "yes" : "no");
	}
}

int main() {
	/* Open the first available device */
	printf("Opening first available device...\n");
	icsneoc2_device_t* device = NULL;
	icsneoc2_error_t res = icsneoc2_device_open_first(0, icsneoc2_open_options_default, &device);
	if(res != icsneoc2_error_success) {
		return print_error_code("\tFailed to open first device", res);
	}

	char description[255] = {0};
	size_t description_length = 255;
	res = icsneoc2_device_description_get(device, description, &description_length);
	if(res != icsneoc2_error_success) {
		icsneoc2_device_close(device);
		return print_error_code("\tFailed to get device description", res);
	}
	printf("\tOpened device: %s\n\n", description);

	/* Configure LIN settings */
	int64_t baud = 19200;

	printf("Enable LIN 01 commander resistor... ");
	res = icsneoc2_settings_commander_resistor_set(device, icsneoc2_netid_lin_01, true);
	printf("%s\n", res == icsneoc2_error_success ? "OK" : "FAIL");

	printf("Disable LIN 02 commander resistor... ");
	res = icsneoc2_settings_commander_resistor_set(device, icsneoc2_netid_lin_02, false);
	printf("%s\n", res == icsneoc2_error_success ? "OK" : "FAIL");

	printf("Setting LIN 01 baudrate to %" PRId64 " bit/s... ", baud);
	res = icsneoc2_settings_baudrate_set(device, icsneoc2_netid_lin_01, baud);
	printf("%s\n", res == icsneoc2_error_success ? "OK" : "FAIL");

	printf("Setting LIN 02 baudrate to %" PRId64 " bit/s... ", baud);
	res = icsneoc2_settings_baudrate_set(device, icsneoc2_netid_lin_02, baud);
	printf("%s\n", res == icsneoc2_error_success ? "OK" : "FAIL");

	printf("Setting LIN 01 mode to NORMAL... ");
	res = icsneoc2_settings_lin_mode_set(device, icsneoc2_netid_lin_01, icsneoc2_lin_mode_normal);
	printf("%s\n", res == icsneoc2_error_success ? "OK" : "FAIL");

	printf("Setting LIN 02 mode to NORMAL... ");
	res = icsneoc2_settings_lin_mode_set(device, icsneoc2_netid_lin_02, icsneoc2_lin_mode_normal);
	printf("%s\n", res == icsneoc2_error_success ? "OK" : "FAIL");

	printf("Applying settings... ");
	res = icsneoc2_settings_apply(device);
	if(res != icsneoc2_error_success) {
		print_events(description);
		icsneoc2_device_close(device);
		return print_error_code("\tFailed to apply settings", res);
	}
	printf("OK\n");

	printf("Getting LIN 01 baudrate... ");
	int64_t read_baud = 0;
	res = icsneoc2_settings_baudrate_get(device, icsneoc2_netid_lin_01, &read_baud);
	if(res == icsneoc2_error_success)
		printf("OK, %" PRId64 " bit/s\n", read_baud);
	else
		printf("FAIL\n");

	printf("Getting LIN 02 baudrate... ");
	res = icsneoc2_settings_baudrate_get(device, icsneoc2_netid_lin_02, &read_baud);
	if(res == icsneoc2_error_success)
		printf("OK, %" PRId64 " bit/s\n\n", read_baud);
	else
		printf("FAIL\n\n");

	/* Transmit a LIN responder data update on LIN 02 */
	printf("Transmitting a LIN 02 responder data frame... ");
	{
		icsneoc2_message_t* msg = NULL;
		res = icsneoc2_message_lin_create(&msg, 0x11);
		if(res != icsneoc2_error_success) {
			print_events(description);
			icsneoc2_device_close(device);
			return print_error_code("\tFailed to create LIN message", res);
		}
		icsneoc2_lin_msg_type_t msg_type = icsneoc2_lin_msg_type_update_responder;
		res = icsneoc2_message_lin_props_set(msg, NULL, NULL, &msg_type, NULL);
		uint8_t data[] = {0xaa, 0xbb, 0xcc, 0xdd, 0x11, 0x22, 0x33, 0x44};
		res += icsneoc2_message_data_set(msg, data, sizeof(data));
		res += icsneoc2_message_netid_set(msg, icsneoc2_netid_lin_02);
        res += icsneoc2_message_lin_calc_checksum(msg);
		if(res != icsneoc2_error_success) {
			icsneoc2_message_free(msg);
			print_events(description);
			icsneoc2_device_close(device);
			return print_error_code("\tFailed to set LIN message properties", res);
		}
		res = icsneoc2_device_message_transmit(device, msg);
		icsneoc2_message_free(msg);
		if(res != icsneoc2_error_success) {
			print_events(description);
			icsneoc2_device_close(device);
			return print_error_code("\tFailed to transmit LIN responder message", res);
		}
		printf("OK\n");
	}

	/* Transmit a LIN commander header on LIN 01 */
	printf("Transmitting a LIN 01 commander header... ");
	{
		icsneoc2_message_t* msg = NULL;
		res = icsneoc2_message_lin_create(&msg, 0x11);
		if(res != icsneoc2_error_success) {
			print_events(description);
			icsneoc2_device_close(device);
			return print_error_code("\tFailed to create LIN message", res);
		}
		icsneoc2_lin_msg_type_t msg_type = icsneoc2_lin_msg_type_header_only;
		res = icsneoc2_message_lin_props_set(msg, NULL, NULL, &msg_type, NULL);
		res += icsneoc2_message_netid_set(msg, icsneoc2_netid_lin_01);
		if(res != icsneoc2_error_success) {
			icsneoc2_message_free(msg);
			print_events(description);
			icsneoc2_device_close(device);
			return print_error_code("\tFailed to set LIN message properties", res);
		}
		res = icsneoc2_device_message_transmit(device, msg);
		icsneoc2_message_free(msg);
		if(res != icsneoc2_error_success) {
			print_events(description);
			icsneoc2_device_close(device);
			return print_error_code("\tFailed to transmit LIN header", res);
		}
		printf("OK\n\n");
	}

	sleep_ms(100);

	/* Transmit a LIN commander message with data on LIN 01 */
	printf("Transmitting a LIN 01 commander frame with data... ");
	{
		icsneoc2_message_t* msg = NULL;
		res = icsneoc2_message_lin_create(&msg, 0x22);
		if(res != icsneoc2_error_success) {
			print_events(description);
			icsneoc2_device_close(device);
			return print_error_code("\tFailed to create LIN message", res);
		}
		icsneoc2_lin_msg_type_t msg_type = icsneoc2_lin_msg_type_commander_msg;
		bool enhanced = true;
		res = icsneoc2_message_lin_props_set(msg, NULL, NULL, &msg_type, &enhanced);
		uint8_t data[] = {0x11, 0x22, 0x33, 0x44, 0xaa, 0xbb, 0xcc, 0xdd};
		res += icsneoc2_message_data_set(msg, data, sizeof(data));
		res += icsneoc2_message_netid_set(msg, icsneoc2_netid_lin_01);
        res += icsneoc2_message_lin_calc_checksum(msg);
		if(res != icsneoc2_error_success) {
			icsneoc2_message_free(msg);
			print_events(description);
			icsneoc2_device_close(device);
			return print_error_code("\tFailed to set LIN message properties", res);
		}
		res = icsneoc2_device_message_transmit(device, msg);
		icsneoc2_message_free(msg);
		if(res != icsneoc2_error_success) {
			print_events(description);
			icsneoc2_device_close(device);
			return print_error_code("\tFailed to transmit LIN commander message", res);
		}
		printf("OK\n\n");
	}

	sleep_ms(100);

	/* Read back any received messages and display LIN frames */
	icsneoc2_message_t* messages[2048] = {0};
	size_t message_count = 2048;
	printf("Getting messages...\n");
	for(size_t i = 0; i < message_count; ++i) {
		res = icsneoc2_device_message_get(device, &messages[i], 0);
		if(res != icsneoc2_error_success) {
			for(size_t j = 0; j < i; ++j) {
				icsneoc2_message_free(messages[j]);
			}
			print_events(description);
			icsneoc2_device_close(device);
			return print_error_code("\tFailed to get messages", res);
		}
		if(messages[i] == NULL) {
			message_count = i;
			break;
		}
	}
	printf("\tReceived %zu messages\n", message_count);
	process_lin_messages(messages, message_count);
	for(size_t i = 0; i < message_count; ++i) {
		icsneoc2_message_free(messages[i]);
	}

	/* Cleanup */
	print_events(description);
	printf("Closing device... ");
	res = icsneoc2_device_close(device);
	printf("%s\n", res == icsneoc2_error_success ? "OK" : "FAIL");

	return 0;
}

LIN Transmit

Download example

#include <icsneo/icsneoc2.h>
#include <icsneo/icsneoc2messages.h>
#include <icsneo/icsneoc2settings.h>

#ifdef _WIN32
#include <windows.h>
#else
#include <unistd.h>
#endif

#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <inttypes.h>

void sleep_ms(uint32_t ms) {
#ifdef _WIN32
	Sleep(ms);
#else
	sleep(ms / 1000);
#endif
}

int print_error_code(const char* message, icsneoc2_error_t error) {
	char error_str[64];
	size_t error_str_len = sizeof(error_str);
	icsneoc2_error_t res = icsneoc2_error_code_get(error, error_str, &error_str_len);
	if(res != icsneoc2_error_success) {
		printf("%s: Failed to get string for error code %d with error code %d\n", message, error, res);
		return res;
	}
	printf("%s: \"%s\" (%u)\n", message, error_str, error);
	return (int)error;
}

int read_int(const char* prompt, int min_val, int max_val) {
	int value;
	while(1) {
		printf("%s", prompt);
		if(scanf("%d", &value) != 1) {
			/* Clear invalid input */
			int c;
			while((c = getchar()) != '\n' && c != EOF) {}
			printf("Invalid input, try again.\n");
			continue;
		}
		if(value < min_val || value > max_val) {
			printf("Please enter a value between %d and %d.\n", min_val, max_val);
			continue;
		}
		return value;
	}
}

int read_hex(const char* prompt, int min_val, int max_val) {
	int value;
	while(1) {
		printf("%s", prompt);
		if(scanf("%x", &value) != 1) {
			/* Clear invalid input */
			int c;
			while((c = getchar()) != '\n' && c != EOF) {}
			printf("Invalid input, try again.\n");
			continue;
		}
		if(value < min_val || value > max_val) {
			printf("Please enter a value between 0x%X and 0x%X.\n", min_val, max_val);
			continue;
		}
		return value;
	}
}

int main() {
	icsneoc2_error_t res;

	/* ===== Device Selection ===== */
	printf("Searching for devices...\n");
	icsneoc2_device_info_t* found_devices = NULL;
	res = icsneoc2_device_enumerate(0, &found_devices);
	if(res != icsneoc2_error_success) {
		return print_error_code("Failed to enumerate devices", res);
	}
	if(!found_devices) {
		printf("No devices found.\n");
		return 1;
	}

	/* Count and display devices */
	int device_count = 0;
	for(icsneoc2_device_info_t* cur = found_devices; cur; cur = icsneoc2_device_info_next(cur)) {
		char desc[128] = {0};
		size_t desc_len = sizeof(desc);
		icsneoc2_device_info_description_get(cur, desc, &desc_len);
		printf("  [%d] %s\n", device_count + 1, desc);
		device_count++;
	}

	int device_choice = read_int("Select device: ", 1, device_count);

	/* Find the selected device_info node */
	icsneoc2_device_info_t* selected_info = found_devices;
	for(int i = 1; i < device_choice; i++) {
		selected_info = icsneoc2_device_info_next(selected_info);
	}

	/* Open the selected device */
	icsneoc2_device_t* device = NULL;
	res = icsneoc2_device_create(selected_info, &device);
	if(res != icsneoc2_error_success) {
		icsneoc2_enumeration_free(found_devices);
		return print_error_code("Failed to create device from device info", res);
	}
	res = icsneoc2_device_open(device, icsneoc2_open_options_default);
	icsneoc2_enumeration_free(found_devices);
	if(res != icsneoc2_error_success) {
		icsneoc2_device_free(device);
		return print_error_code("Failed to open device", res);
	}

	char description[128] = {0};
	size_t description_length = sizeof(description);
	icsneoc2_device_description_get(device, description, &description_length);
	printf("Opened device: %s\n\n", description);

	/* ===== LIN Network Selection ===== */
	/* Get all supported TX networks */
	size_t tx_net_count = 0;
	res = icsneoc2_device_supported_tx_networks_get(device, NULL, &tx_net_count);
	if(res != icsneoc2_error_success || tx_net_count == 0) {
		printf("No supported TX networks.\n");
		icsneoc2_device_close(device);
		return 1;
	}

	icsneoc2_netid_t* tx_networks = (icsneoc2_netid_t*)malloc(tx_net_count * sizeof(icsneoc2_netid_t));
	if(!tx_networks) {
		printf("Out of memory.\n");
		icsneoc2_device_close(device);
		icsneoc2_device_free(device);
		return 1;
	}
	res = icsneoc2_device_supported_tx_networks_get(device, tx_networks, &tx_net_count);
	if(res != icsneoc2_error_success) {
		free(tx_networks);
		icsneoc2_device_close(device);
		icsneoc2_device_free(device);
		return print_error_code("Failed to get TX networks", res);
	}

	/* Filter for LIN networks */
	icsneoc2_netid_t lin_networks[64] = {0};
	char lin_names[64][128] = {{0}};
	size_t lin_count = 0;
	for(size_t i = 0; i < tx_net_count && lin_count < 64; i++) {
		icsneoc2_message_t* tmp = NULL;
		res = icsneoc2_message_lin_create(&tmp, 0);
		if(res != icsneoc2_error_success) continue;
		res = icsneoc2_message_netid_set(tmp, tx_networks[i]);
		if(res != icsneoc2_error_success) { icsneoc2_message_free(tmp); continue; }
		icsneoc2_network_type_t ntype = 0;
		res = icsneoc2_message_network_type_get(tmp, &ntype);
		icsneoc2_message_free(tmp);
		if(res != icsneoc2_error_success) continue;
		if(ntype == icsneoc2_network_type_lin) {
			lin_networks[lin_count] = tx_networks[i];
			size_t name_len = 128;
			icsneoc2_netid_name_get(tx_networks[i], lin_names[lin_count], &name_len);
			lin_count++;
		}
	}
	free(tx_networks);

	if(lin_count == 0) {
		printf("No LIN networks available on this device.\n");
		icsneoc2_device_close(device);
		icsneoc2_device_free(device);
		return 1;
	}

	printf("Available LIN networks:\n");
	for(size_t i = 0; i < lin_count; i++) {
		printf("  [%zu] %s\n", i + 1, lin_names[i]);
	}

	int lin_choice = read_int("Select LIN network: ", 1, (int)lin_count);
	icsneoc2_netid_t selected_netid = lin_networks[lin_choice - 1];
	printf("Selected: %s\n\n", lin_names[lin_choice - 1]);

	/* ===== Commander / Responder Selection ===== */
	printf("Message type:\n");
	printf("  [1] Commander frame\n");
	printf("  [2] Responder frame (update responder + header only)\n");
	int type_choice = read_int("Select message type: ", 1, 2);
	bool is_commander = (type_choice == 1);
	printf("Selected: %s\n\n", is_commander ? "Commander" : "Responder");

	uint8_t id_choice = (uint8_t)read_hex("Select LIN ID (0x00-0x3F): ", 0, 0x3F);
	printf("Selected: 0x%02X\n\n", id_choice);

	/* ===== Configure LIN ===== */
	printf("Configuring %s... ", lin_names[lin_choice - 1]);
	res = icsneoc2_settings_commander_resistor_set(device, selected_netid, is_commander);
	res += icsneoc2_settings_baudrate_set(device, selected_netid, 19200);
	res += icsneoc2_settings_lin_mode_set(device, selected_netid, icsneoc2_lin_mode_normal);
	res += icsneoc2_settings_apply(device);
	if(res != icsneoc2_error_success) {
		icsneoc2_device_close(device);
		icsneoc2_device_free(device);
		return print_error_code("Failed to configure LIN", res);
	}
	printf("OK\n\n");

	/* ===== Transmit Loop ===== */
	printf("Transmitting on %s every second for 10 seconds...\n", lin_names[lin_choice - 1]);
	uint8_t counter = 0;
	for(int i = 0; i < 10; i++) {
		icsneoc2_message_t* msg = NULL;
		res = icsneoc2_message_lin_create(&msg, id_choice);
		if(res != icsneoc2_error_success) {
			print_error_code("\tFailed to create LIN message", res);
			break;
		}

		uint8_t data[] = {counter, counter + 1, counter + 2, counter + 3, 0xAA, 0xBB, 0xCC, 0xDD};
		bool enhanced = true;

		if(is_commander) {
			/* Commander: send header-only frame to poll the bus */
			icsneoc2_lin_msg_type_t msg_type = icsneoc2_lin_msg_type_header_only;
			res = icsneoc2_message_lin_props_set(msg, NULL, NULL, &msg_type, &enhanced);
			res += icsneoc2_message_netid_set(msg, selected_netid);
			if(res != icsneoc2_error_success) {
				icsneoc2_message_free(msg);
				print_error_code("\tFailed to set commander properties", res);
				break;
			}
			res = icsneoc2_device_message_transmit(device, msg);
			icsneoc2_message_free(msg);
			if(res != icsneoc2_error_success) {
				print_error_code("\tFailed to transmit header", res);
				break;
			}
		} else {
			/* Responder: update the responder table with new data */
			icsneoc2_lin_msg_type_t msg_type = icsneoc2_lin_msg_type_update_responder;
			res = icsneoc2_message_lin_props_set(msg, NULL, NULL, &msg_type, &enhanced);
			res += icsneoc2_message_data_set(msg, data, sizeof(data));
			res += icsneoc2_message_lin_calc_checksum(msg);
			res += icsneoc2_message_netid_set(msg, selected_netid);
			if(res != icsneoc2_error_success) {
				icsneoc2_message_free(msg);
				print_error_code("\tFailed to update responder", res);
				break;
			}
			res = icsneoc2_device_message_transmit(device, msg);
			icsneoc2_message_free(msg);
			if(res != icsneoc2_error_success) {
				print_error_code("\tFailed to transmit responder update", res);
				break;
			}
		}
		printf("[%2d/10] Transmitted %s msg ID=0x%02X, counter=%u\n",
			i + 1, is_commander ? "commander" : "responder", id_choice, counter);
		counter += 4;
		sleep_ms(1000);
	}

	/* Cleanup */
	printf("\nClosing device... ");
	res = icsneoc2_device_close(device);
	printf("%s\n", res == icsneoc2_error_success ? "OK" : "FAIL");
	icsneoc2_device_free(device);

	return 0;
}

Ethernet Transmit

Download example

#include <icsneo/icsneoc2.h>
#include <icsneo/icsneoc2messages.h>

#include <stdio.h>
#include <inttypes.h>

#ifdef _WIN32
#include <windows.h>
#else
#include <unistd.h>
#endif

int print_error_code(const char* message, icsneoc2_error_t error) {
	char error_str[64];
	size_t error_str_len = sizeof(error_str);
	icsneoc2_error_t res = icsneoc2_error_code_get(error, error_str, &error_str_len);
	if(res != icsneoc2_error_success) {
		printf("%s: Failed to get string for error code %d with error code %d\n", message, error, res);
		return res;
	}
	printf("%s: \"%s\" (%u)\n", message, error_str, error);
	return (int)error;
}

void print_events(void) {
	icsneoc2_event_t* events[256] = {0};
	size_t events_count = 256;
	for(size_t i = 0; i < events_count; ++i) {
		icsneoc2_error_t res = icsneoc2_event_get(&events[i], NULL);
		if(res != icsneoc2_error_success) {
			(void)print_error_code("\tFailed to get events", res);
			return;
		}
		if(events[i] == NULL) {
			events_count = i;
			break;
		}
	}
	for(size_t i = 0; i < events_count; i++) {
		char description[255] = {0};
		size_t description_length = 255;
		icsneoc2_error_t res = icsneoc2_event_description_get(events[i], description, &description_length);
		if(res != icsneoc2_error_success) {
			print_error_code("\tFailed to get event description", res);
			continue;
		}
		printf("\tEvent %zu: %s\n", i, description);
	}
	for(size_t i = 0; i < events_count; i++) {
		icsneoc2_event_free(events[i]);
	}
	if(events_count > 0) {
		printf("\tReceived %zu events\n", events_count);
	}
}

int main(void) {
	/* Open the first available device */
	printf("Opening first available device...\n");
	icsneoc2_device_t* device = NULL;
	icsneoc2_error_t res = icsneoc2_device_open_first(0, icsneoc2_open_options_default, &device);
	if(res != icsneoc2_error_success) {
		return print_error_code("Failed to open first device", res);
	}

	/* Get a description of the opened device */
	char description[255] = {0};
	size_t description_length = 255;
	res = icsneoc2_device_description_get(device, description, &description_length);
	if(res != icsneoc2_error_success) {
		return print_error_code("Failed to get device description", res);
	}
	printf("Opened device: %s\n", description);

	icsneoc2_netid_t tx_networks[255] = {0};
    size_t tx_net_count = sizeof(tx_networks) / sizeof(tx_networks[0]);
	res = icsneoc2_device_supported_tx_networks_get(device, tx_networks, &tx_net_count);
	if(res != icsneoc2_error_success) {
		print_events();
		return print_error_code("Failed to get TX networks", res);
	}

	/* Filter for Ethernet/AutomotiveEthernet networks */
	icsneoc2_netid_t eth_networks[64] = {0};
	char eth_names[64][128] = {{0}};
	size_t eth_count = 0;
	for(size_t i = 0; i < tx_net_count && eth_count < 64; i++) {
		/* Create a temporary message to check network type */
		icsneoc2_message_t* tmp = NULL;
		res = icsneoc2_message_eth_create(&tmp);
		if(res != icsneoc2_error_success) continue;
		res = icsneoc2_message_netid_set(tmp, tx_networks[i]);
		if(res != icsneoc2_error_success) { icsneoc2_message_free(tmp); continue; }
		icsneoc2_network_type_t ntype = 0;
		res = icsneoc2_message_network_type_get(tmp, &ntype);
		icsneoc2_message_free(tmp);
		if(res != icsneoc2_error_success) continue;
		if(ntype == icsneoc2_network_type_ethernet || ntype == icsneoc2_network_type_automotive_ethernet) {
			eth_networks[eth_count] = tx_networks[i];
			size_t name_len = 128;
			icsneoc2_netid_name_get(tx_networks[i], eth_names[eth_count], &name_len);
			eth_count++;
		}
	}

	if(eth_count == 0) {
		printf("No Ethernet TX networks available on this device.\n");
		icsneoc2_device_close(device);
		return 0;
	}

	/* Let the user pick */
	printf("Available Ethernet TX networks:\n");
	for(size_t i = 0; i < eth_count; i++) {
		printf("  %zu) %s\n", i + 1, eth_names[i]);
	}
	printf("Select network [1-%zu]: ", eth_count);
	int selection = 0;
	if(scanf("%d", &selection) != 1 || selection < 1 || (size_t)selection > eth_count) {
		printf("Invalid selection, using first available.\n");
		selection = 1;
	}
	icsneoc2_netid_t netid = eth_networks[selection - 1];
	printf("Selected: %s\n", eth_names[selection - 1]);
	/* Transmit Ethernet frames */
	const size_t msg_count = 10;
	printf("Transmitting %zu Ethernet frames on %s...\n", msg_count, eth_names[selection - 1]);

	for(size_t i = 0; i < msg_count; i++) {
		/* Create an Ethernet message */
		icsneoc2_message_t* message = NULL;
		res = icsneoc2_message_eth_create(&message);
		if(res != icsneoc2_error_success) {
			print_events();
			return print_error_code("Failed to create Ethernet message", res);
		}

		/* Set the network ID */
		res = icsneoc2_message_netid_set(message, netid );
		if(res != icsneoc2_error_success) {
			icsneoc2_message_free(message);
			print_events();
			return print_error_code("Failed to set netid", res);
		}

		/* Build Ethernet frame data:
		 * Bytes 0-5:   Destination MAC (00:FC:70:00:01:02)
		 * Bytes 6-11:  Source MAC      (00:FC:70:00:01:01)
		 * Bytes 12-13: EtherType       (0x0800 = IPv4)
		 * Bytes 14+:   Payload
		 */
		uint8_t frame_data[] = {
			0x00, 0xFC, 0x70, 0x00, 0x01, 0x02, /* Destination MAC */
			0x00, 0xFC, 0x70, 0x00, 0x01, 0x01, /* Source MAC */
			0x08, 0x00,                           /* EtherType (IPv4) */
			0x45, 0x00, 0x00, 0x20,               /* IPv4: ver/IHL, DSCP, total length (32) */
			0x00, 0x00, 0x00, 0x00,               /* Identification, flags/fragment offset */
			0x40, 0x11, 0x00, 0x00,               /* TTL (64), protocol (UDP), checksum (0) */
			0xC0, 0xA8, 0x01, 0x01,               /* Source IP (192.168.1.1) */
			0xC0, 0xA8, 0x01, 0x02,               /* Destination IP (192.168.1.2) */
			0xC3, 0x50, 0xC3, 0x51,               /* UDP: src port (50000), dst port (50001) */
			0x00, 0x0C, 0x00, 0x00,               /* UDP: length (12), checksum (0) */
			0x00, 0x00, 0x00, 0x00                /* UDP payload (4 bytes, frame counter) */
		};

		/* Put the frame counter in the UDP payload */
		frame_data[42] = (uint8_t)((i >> 24) & 0xFF);
		frame_data[43] = (uint8_t)((i >> 16) & 0xFF);
		frame_data[44] = (uint8_t)((i >> 8) & 0xFF);
		frame_data[45] = (uint8_t)(i & 0xFF);

		res = icsneoc2_message_data_set(message, frame_data, sizeof(frame_data));
		if(res != icsneoc2_error_success) {
			icsneoc2_message_free(message);
			print_events();
			return print_error_code("Failed to set frame data", res);
		}

		/* Transmit the message */
		res = icsneoc2_device_message_transmit(device, message);
		if(res != icsneoc2_error_success) {
			icsneoc2_message_free(message);
			print_events();
			return print_error_code("Failed to transmit Ethernet frame", res);
		}

		icsneoc2_message_free(message);
		printf("\tTransmitted frame %zu\n", i + 1);
	}

	printf("Successfully transmitted %zu Ethernet frames\n", msg_count);

	/* Print any events */
	print_events();

	/* Close the device */
	printf("Closing device...\n");
	res = icsneoc2_device_close(device);
	if(res != icsneoc2_error_success) {
		return print_error_code("Failed to close device", res);
	}

	return 0;
}

Ethernet Receive

Download example

#include <icsneo/icsneoc2.h>
#include <icsneo/icsneoc2messages.h>

#include <stdio.h>
#include <inttypes.h>

#ifdef _WIN32
#include <windows.h>
#else
#include <unistd.h>
#endif

void sleep_ms(uint32_t ms) {
#ifdef _WIN32
	Sleep(ms);
#else
	sleep(ms / 1000);
#endif
}

int print_error_code(const char* message, icsneoc2_error_t error) {
	char error_str[64];
	size_t error_str_len = sizeof(error_str);
	icsneoc2_error_t res = icsneoc2_error_code_get(error, error_str, &error_str_len);
	if(res != icsneoc2_error_success) {
		printf("%s: Failed to get string for error code %d with error code %d\n", message, error, res);
		return res;
	}
	printf("%s: \"%s\" (%u)\n", message, error_str, error);
	return (int)error;
}

void print_events(void) {
	icsneoc2_event_t* events[256] = {0};
	size_t events_count = 256;
	for(size_t i = 0; i < events_count; ++i) {
		icsneoc2_error_t res = icsneoc2_event_get(&events[i], NULL);
		if(res != icsneoc2_error_success) {
			(void)print_error_code("\tFailed to get events", res);
			return;
		}
		if(events[i] == NULL) {
			events_count = i;
			break;
		}
	}
	for(size_t i = 0; i < events_count; i++) {
		char description[255] = {0};
		size_t description_length = 255;
		icsneoc2_error_t res = icsneoc2_event_description_get(events[i], description, &description_length);
		if(res != icsneoc2_error_success) {
			print_error_code("\tFailed to get event description", res);
			continue;
		}
		printf("\tEvent %zu: %s\n", i, description);
	}
	for(size_t i = 0; i < events_count; i++) {
		icsneoc2_event_free(events[i]);
	}
	if(events_count > 0) {
		printf("\tReceived %zu events\n", events_count);
	}
}

void print_mac(const char* label, const uint8_t* mac) {
	printf("%s: %02x:%02x:%02x:%02x:%02x:%02x", label, mac[0], mac[1], mac[2], mac[3], mac[4], mac[5]);
}

int process_ethernet_message(icsneoc2_message_t* message, size_t index) {
	icsneoc2_netid_t netid = 0;
	char netid_name[128] = {0};
	size_t netid_name_length = 128;

	icsneoc2_error_t res = icsneoc2_message_netid_get(message, &netid);
	if(res != icsneoc2_error_success) {
		return print_error_code("\tFailed to get netid", res);
	}
	res = icsneoc2_netid_name_get(netid, netid_name, &netid_name_length);
	if(res != icsneoc2_error_success) {
		return print_error_code("\tFailed to get netid name", res);
	}

	/* Get data length first */
	size_t data_length = 0;
	res = icsneoc2_message_data_get(message, NULL, &data_length);
	if(res != icsneoc2_error_success) {
		return print_error_code("\tFailed to get data length", res);
	}

	printf("\t%zu) Ethernet Frame on %s (0x%x) - %zu bytes\n", index, netid_name, netid, data_length);

	/* Get MAC addresses and EtherType if we have enough data */
	uint8_t dst_mac[6] = {0};
	uint8_t src_mac[6] = {0};
	uint16_t ether_type = 0;

	res = icsneoc2_message_eth_mac_get(message, dst_mac, src_mac);
	if(res == icsneoc2_error_success) {
		printf("\t  ");
		print_mac("Dst", dst_mac);
		printf("  ");
		print_mac("Src", src_mac);
		printf("\n");
	}

	res = icsneoc2_message_eth_ether_type_get(message, &ether_type);
	if(res == icsneoc2_error_success) {
		printf("\t  EtherType: 0x%04x\n", ether_type);
	}

	/* Get flags */
	icsneoc2_message_eth_flags_t flags = 0;
	res = icsneoc2_message_eth_props_get(message, &flags, NULL, NULL);
	if(res == icsneoc2_error_success && flags != 0) {
		printf("\t  Flags: 0x%" PRIx64, flags);
		if(flags & ICSNEOC2_MESSAGE_ETH_FLAGS_CRC_ERROR) printf(" [CRC_ERROR]");
		if(flags & ICSNEOC2_MESSAGE_ETH_FLAGS_FRAME_TOO_SHORT) printf(" [FRAME_TOO_SHORT]");
		if(flags & ICSNEOC2_MESSAGE_ETH_FLAGS_TX_ABORTED) printf(" [TX_ABORTED]");
		if(flags & ICSNEOC2_MESSAGE_ETH_FLAGS_FCS_VERIFIED) printf(" [FCS_VERIFIED]");
		if(flags & ICSNEOC2_MESSAGE_ETH_FLAGS_PREEMPTION_ENABLED) printf(" [PREEMPTION]");
		if(flags & ICSNEOC2_MESSAGE_ETH_FLAGS_IS_T1S) printf(" [T1S]");
		printf("\n");
	}

	/* Print data bytes */
	uint8_t data[1600] = {0};
	res = icsneoc2_message_data_get(message, data, &data_length);
	if(res == icsneoc2_error_success) {
		printf("\t  Data:\n\t    ");
		for(size_t x = 0; x < data_length; x++) {
			printf("0x%02x ", data[x]);
			if((x + 1) % 20 == 0 && x + 1 < data_length) {
				printf("\n\t    ");
			}
		}
		printf("\n");
	}

	return icsneoc2_error_success;
}

int main(void) {
	/* Open the first available device */
	printf("Opening first available device...\n");
	icsneoc2_device_t* device = NULL;
	icsneoc2_error_t res = icsneoc2_device_open_first(0, icsneoc2_open_options_default, &device);
	if(res != icsneoc2_error_success) {
		return print_error_code("Failed to open first device", res);
	}

	/* Get a description of the opened device */
	char description[255] = {0};
	size_t description_length = 255;
	res = icsneoc2_device_description_get(device, description, &description_length);
	if(res != icsneoc2_error_success) {
		return print_error_code("Failed to get device description", res);
	}
	printf("Opened device: %s\n", description);

	/* Wait for Ethernet frames to arrive */
	const int duration_seconds = 10;
	printf("Listening for Ethernet frames for %d seconds...\n", duration_seconds);
	sleep_ms(duration_seconds * 1000);

	/* Retrieve and process messages */
	icsneoc2_message_t* messages[20000] = {0};
	size_t message_count = 20000;
	size_t eth_count = 0;

	for(size_t i = 0; i < message_count; ++i) {
		res = icsneoc2_device_message_get(device, &messages[i], 0);
		if(res != icsneoc2_error_success) {
			print_events();
			return print_error_code("Failed to get messages", res);
		}
		if(messages[i] == NULL) {
			message_count = i;
			break;
		}
	}

	printf("Got %zu messages total, filtering for Ethernet...\n", message_count);

	for(size_t i = 0; i < message_count; i++) {
		icsneoc2_message_t* message = messages[i];

		/* Check if this is a TX echo (skip it) */
		bool is_tx = false;
		res = icsneoc2_message_is_transmit(message, &is_tx);
		if(res != icsneoc2_error_success || is_tx) {
			continue;
		}

		/* Check if this is an Ethernet message */
		bool is_ethernet = false;
		res = icsneoc2_message_is_ethernet(message, &is_ethernet);
		if(res != icsneoc2_error_success || !is_ethernet) {
			continue;
		}

		process_ethernet_message(message, eth_count);
		eth_count++;
	}

	printf("Received %zu Ethernet frames out of %zu total messages\n", eth_count, message_count);

	/* Free all messages */
	for(size_t i = 0; i < message_count; ++i) {
		icsneoc2_message_free(messages[i]);
	}

	/* Print any events */
	print_events();

	/* Close the device */
	printf("Closing device...\n");
	res = icsneoc2_device_close(device);
	if(res != icsneoc2_error_success) {
		return print_error_code("Failed to close device", res);
	}

	return 0;
}

T1S Loopback

Download example

#include <icsneo/icsneoc2.h>
#include <icsneo/icsneoc2messages.h>
#include <icsneo/icsneoc2settings.h>

#include <inttypes.h>
#include <stdbool.h>
#include <stdlib.h>
#include <stdio.h>
#include <string.h>

#ifdef _WIN32
#include <windows.h>
#else
#include <unistd.h>
#endif

#define TX_LOCAL_ID 0u
#define RX_LOCAL_ID 1u
#define T1S_MAX_NODES 8u
#define T1S_TX_OPP_TIMER 20u
#define T1S_BURST_TIMER 64u
#define T1S_MAX_BURST 128u
#define LOOPBACK_ETHER_TYPE 0x9000u
#define LOOPBACK_FRAME_SIZE 60u

typedef struct selectable_network {
	icsneoc2_netid_t netid;
	char name[64];
} selectable_network_t;

/* Sleep for a short period while waiting for the device to apply settings. */
static void sleep_ms(uint32_t ms);

/* Print a readable error string and return the same failure code to the caller. */
static int print_error_code(const char* message, icsneoc2_error_t error);

/* Drain and print queued library events when the example encounters an error. */
static void print_events(void);

/* Convert a netid to a readable name such as "AE 02". */
static int get_netid_name(icsneoc2_netid_t netid, char* buffer, size_t buffer_size);

/* Gather the device's TX and RX networks, keeping only automotive Ethernet ports. */
static int get_available_networks(const icsneoc2_device_t* device,
		selectable_network_t* tx_networks,
		size_t* tx_count,
		selectable_network_t* rx_networks,
		size_t* rx_count);

/* Prompt the user to choose one TX or RX network from the filtered list. */
static int prompt_for_network_selection(const char* label, const selectable_network_t* networks, size_t count, icsneoc2_netid_t* selected_netid, char* selected_name, size_t selected_name_size);

/* Apply the small set of T1S settings needed for this two-port loopback example. */
static int configure_t1s_port(icsneoc2_device_t* device, icsneoc2_netid_t netid, uint8_t local_id);

/* Print a MAC address in a compact human-readable form. */
static void print_mac(const char* label, const uint8_t* mac);

/* Print payload bytes as hex for the TX echo and RX frame output. */
static void print_payload_hex(const uint8_t* data, size_t length);

/* Build one recognizable Ethernet frame that both transmit and receive paths share. */
static void build_loopback_frame(uint8_t* frame_data, size_t frame_size, const char* tx_name, const char* rx_name);

/* Check whether a received Ethernet frame matches the loopback frame this example sent. */
static int message_matches_loopback_frame(icsneoc2_message_t* message, const uint8_t* expected, size_t expected_length, bool* matches);

/* Print the key details of an Ethernet message found during the loopback test. */
static int print_ethernet_message(icsneoc2_message_t* message, const char* direction_label);

/* Transmit the loopback Ethernet frame on the selected TX port. */
static int transmit_loopback_frame(icsneoc2_device_t* device, icsneoc2_netid_t tx_netid, const char* tx_name, const char* rx_name);

/* Poll until the example sees both the TX echo and the matching RX frame. */
static int poll_for_loopback_messages(icsneoc2_device_t* device,
		icsneoc2_netid_t tx_netid,
		icsneoc2_netid_t rx_netid,
		const char* tx_name,
		const char* rx_name,
		const uint8_t* expected_frame,
		size_t expected_frame_length);

int main(void) {
	icsneoc2_device_t* device = NULL;
	icsneoc2_error_t res;
	char description[255] = {0};
	char serial[64] = {0};
	size_t description_length = sizeof(description);
	size_t serial_length = sizeof(serial);
	selectable_network_t available_tx_networks[128] = {0};
	selectable_network_t available_rx_networks[128] = {0};
	size_t available_tx_count = 0;
	size_t available_rx_count = 0;
	icsneoc2_netid_t tx_netid = 0;
	icsneoc2_netid_t rx_netid = 0;
	uint8_t expected_frame[LOOPBACK_FRAME_SIZE] = {0};
	char tx_name[64] = {0};
	char rx_name[64] = {0};
	int status = 1;

	printf("RAD-Comet3 C2 T1S loopback example\n");
	printf("Opening first available RAD-Comet3...\n");

	res = icsneoc2_device_open_first(icsneoc2_devicetype_rad_comet3, icsneoc2_open_options_default, &device);
	if(res != icsneoc2_error_success) {
		return print_error_code("Failed to open a RAD-Comet3", res);
	}

	res = icsneoc2_device_description_get(device, description, &description_length);
	if(res != icsneoc2_error_success) {
		print_error_code("Failed to get device description", res);
		goto cleanup;
	}
	res = icsneoc2_device_serial_get(device, serial, &serial_length);
	if(res != icsneoc2_error_success) {
		print_error_code("Failed to get device serial", res);
		goto cleanup;
	}
	printf("Opened device: %s [%s]\n", description, serial);

	if(get_available_networks(device, available_tx_networks, &available_tx_count, available_rx_networks, &available_rx_count) != 0) {
		goto cleanup;
	}

	if(prompt_for_network_selection("TX", available_tx_networks, available_tx_count, &tx_netid, tx_name, sizeof(tx_name)) != 0) {
		goto cleanup;
	}
	if(prompt_for_network_selection("RX", available_rx_networks, available_rx_count, &rx_netid, rx_name, sizeof(rx_name)) != 0) {
		goto cleanup;
	}

	printf("Selected loopback wiring: %s connected to %s\n", tx_name, rx_name);

	if(tx_netid == rx_netid) {
		printf("TX and RX networks are the same. This example is intended for a physical loopback between two ports.\n");
		goto cleanup;
	}

	// Use the same expected frame bytes for transmit and for receive-side matching.
	build_loopback_frame(expected_frame, sizeof(expected_frame), tx_name, rx_name);

	res = icsneoc2_settings_refresh(device);
	if(res != icsneoc2_error_success) {
		print_error_code("Failed to refresh device settings", res);
		goto cleanup;
	}

	if(configure_t1s_port(device, tx_netid, TX_LOCAL_ID) != 0) {
		goto cleanup;
	}
	if(configure_t1s_port(device, rx_netid, RX_LOCAL_ID) != 0) {
		goto cleanup;
	}

	printf("Applying T1S settings to the device.\n");
	printf("Note: icsneoc2_settings_apply() persists these settings on the device.\n");
	res = icsneoc2_settings_apply(device);
	if(res != icsneoc2_error_success) {
		print_error_code("Failed to apply T1S settings", res);
		goto cleanup;
	}

	sleep_ms(500);

	if(transmit_loopback_frame(device, tx_netid, tx_name, rx_name) != 0) {
		goto cleanup;
	}

	if(poll_for_loopback_messages(device, tx_netid, rx_netid, tx_name, rx_name, expected_frame, sizeof(expected_frame)) != 0) {
		print_events();
		goto cleanup;
	}

	status = 0;

cleanup:
	if(device != NULL) {
		res = icsneoc2_device_close(device);
		if(res != icsneoc2_error_success) {
			(void)print_error_code("Failed to close device", res);
		}
		res = icsneoc2_device_free(device);
		if(res != icsneoc2_error_success) {
			(void)print_error_code("Failed to free device", res);
		}
	}

	return status;
}

static void sleep_ms(uint32_t ms) {
#ifdef _WIN32
	Sleep(ms);
#else
	usleep(ms * 1000);
#endif
}

static int print_error_code(const char* message, icsneoc2_error_t error) {
	char error_str[64] = {0};
	size_t error_str_len = sizeof(error_str);
	icsneoc2_error_t res = icsneoc2_error_code_get(error, error_str, &error_str_len);
	if(res != icsneoc2_error_success) {
		printf("%s: failed to get string for error code %u with error code %u\n", message, error, res);
		return (int)res;
	}
	printf("%s: \"%s\" (%u)\n", message, error_str, error);
	return (int)error;
}

static void print_events(void) {
	icsneoc2_event_t* events[64] = {0};
	size_t count = sizeof(events) / sizeof(events[0]);

	for(size_t i = 0; i < count; ++i) {
		icsneoc2_error_t res = icsneoc2_event_get(&events[i], NULL);
		if(res != icsneoc2_error_success) {
			(void)print_error_code("Failed to get events", res);
			return;
		}
		if(events[i] == NULL) {
			count = i;
			break;
		}
	}

	for(size_t i = 0; i < count; ++i) {
		char description[255] = {0};
		size_t description_length = sizeof(description);
		icsneoc2_error_t res = icsneoc2_event_description_get(events[i], description, &description_length);
		if(res == icsneoc2_error_success) {
			printf("Event %zu: %s\n", i, description);
		}
		icsneoc2_event_free(events[i]);
	}
}

static int get_netid_name(icsneoc2_netid_t netid, char* buffer, size_t buffer_size) {
	size_t length = buffer_size;
	icsneoc2_error_t res = icsneoc2_netid_name_get(netid, buffer, &length);
	if(res != icsneoc2_error_success) {
		return print_error_code("Failed to get netid name", res);
	}
	return 0;
}

static int get_available_networks(const icsneoc2_device_t* device,
		selectable_network_t* tx_networks,
		size_t* tx_count,
		selectable_network_t* rx_networks,
		size_t* rx_count) {
	icsneoc2_netid_t supported_tx_networks[128] = {0};
	icsneoc2_netid_t supported_rx_networks[128] = {0};
	size_t tx_supported_count = sizeof(supported_tx_networks) / sizeof(supported_tx_networks[0]);
	size_t rx_supported_count = sizeof(supported_rx_networks) / sizeof(supported_rx_networks[0]);
	icsneoc2_message_t* probe = NULL;
	icsneoc2_network_type_t network_type = 0;
	icsneoc2_error_t res = icsneoc2_device_supported_tx_networks_get(device, supported_tx_networks, &tx_supported_count);
	if(res != icsneoc2_error_success) {
		return print_error_code("Failed to get supported TX networks", res);
	}
	res = icsneoc2_device_supported_rx_networks_get(device, supported_rx_networks, &rx_supported_count);
	if(res != icsneoc2_error_success) {
		return print_error_code("Failed to get supported RX networks", res);
	}

	*tx_count = 0;
	*rx_count = 0;

	// Reuse one temporary Ethernet message to classify each netid by network type.
	res = icsneoc2_message_eth_create(&probe);
	if(res != icsneoc2_error_success) {
		return print_error_code("Failed to create temporary Ethernet message", res);
	}

	for(size_t i = 0; i < tx_supported_count; ++i) {
		res = icsneoc2_message_netid_set(probe, supported_tx_networks[i]);
		if(res != icsneoc2_error_success) {
			continue;
		}
		res = icsneoc2_message_network_type_get(probe, &network_type);
		if(res != icsneoc2_error_success || network_type != icsneoc2_network_type_automotive_ethernet) {
			continue;
		}
		tx_networks[*tx_count].netid = supported_tx_networks[i];
		if(get_netid_name(supported_tx_networks[i], tx_networks[*tx_count].name, sizeof(tx_networks[*tx_count].name)) != 0) {
			icsneoc2_message_free(probe);
			return 1;
		}
		(*tx_count)++;
	}

	for(size_t i = 0; i < rx_supported_count; ++i) {
		res = icsneoc2_message_netid_set(probe, supported_rx_networks[i]);
		if(res != icsneoc2_error_success) {
			continue;
		}
		res = icsneoc2_message_network_type_get(probe, &network_type);
		if(res != icsneoc2_error_success || network_type != icsneoc2_network_type_automotive_ethernet) {
			continue;
		}
		rx_networks[*rx_count].netid = supported_rx_networks[i];
		if(get_netid_name(supported_rx_networks[i], rx_networks[*rx_count].name, sizeof(rx_networks[*rx_count].name)) != 0) {
			icsneoc2_message_free(probe);
			return 1;
		}
		(*rx_count)++;
	}
	icsneoc2_message_free(probe);

	if(*tx_count == 0) {
		printf("No automotive Ethernet TX networks are available on this device.\n");
		return 1;
	}
	if(*rx_count == 0) {
		printf("No automotive Ethernet RX networks are available on this device.\n");
		return 1;
	}
	return 0;
}

static int prompt_for_network_selection(const char* label, const selectable_network_t* networks, size_t count, icsneoc2_netid_t* selected_netid, char* selected_name, size_t selected_name_size) {
	char input[32] = {0};
	char* end_ptr = NULL;
	long selected_index = 0;

	if(!label || !networks || count == 0 || !selected_netid || !selected_name || selected_name_size == 0) {
		return print_error_code("Invalid network selection parameters", icsneoc2_error_invalid_parameters);
	}

	printf("Available automotive Ethernet %s networks:\n", label);
	for(size_t i = 0; i < count; ++i) {
		printf("  %zu) %s\n", i + 1, networks[i].name);
	}
	printf("Select %s network [1-%zu, default 1]: ", label, count);

	if(fgets(input, sizeof(input), stdin) == NULL || input[0] == '\n') {
		selected_index = 1;
	} else {
		selected_index = strtol(input, &end_ptr, 10);
		if(end_ptr == input || selected_index < 1 || (size_t)selected_index > count) {
			printf("Invalid selection, using %s.\n", networks[0].name);
			selected_index = 1;
		}
	}

	*selected_netid = networks[selected_index - 1].netid;
	strncpy(selected_name, networks[selected_index - 1].name, selected_name_size - 1);
	selected_name[selected_name_size - 1] = '\0';
	return 0;
}

static int configure_t1s_port(icsneoc2_device_t* device, icsneoc2_netid_t netid, uint8_t local_id) {
	char netid_name[64] = {0};
	icsneoc2_error_t res;
	bool termination = false;

	if(get_netid_name(netid, netid_name, sizeof(netid_name)) != 0) {
		return 1;
	}

	printf("Configuring %s: PLCA on, LocalID=%u, MaxNodes=%u, TxOpp=%u, BurstTimer=%u, MaxBurst=%u\n",
		netid_name,
		(unsigned)local_id,
		(unsigned)T1S_MAX_NODES,
		(unsigned)T1S_TX_OPP_TIMER,
		(unsigned)T1S_BURST_TIMER,
		(unsigned)T1S_MAX_BURST);

	// Keep the example explicit about the small set of T1S settings needed for loopback.
	res = icsneoc2_settings_t1s_plca_enabled_for_set(device, netid, true);
	if(res != icsneoc2_error_success) return print_error_code("Failed to enable T1S PLCA", res);
	res = icsneoc2_settings_t1s_local_id_set(device, netid, local_id);
	if(res != icsneoc2_error_success) return print_error_code("Failed to set T1S local ID", res);
	res = icsneoc2_settings_t1s_max_nodes_set(device, netid, T1S_MAX_NODES);
	if(res != icsneoc2_error_success) return print_error_code("Failed to set T1S max nodes", res);
	res = icsneoc2_settings_t1s_tx_opp_timer_set(device, netid, T1S_TX_OPP_TIMER);
	if(res != icsneoc2_error_success) return print_error_code("Failed to set T1S TX opportunity timer", res);
	res = icsneoc2_settings_t1s_burst_timer_set(device, netid, T1S_BURST_TIMER);
	if(res != icsneoc2_error_success) return print_error_code("Failed to set T1S burst timer", res);
	res = icsneoc2_settings_t1s_max_burst_timer_for_set(device, netid, T1S_MAX_BURST);
	if(res != icsneoc2_error_success) return print_error_code("Failed to set T1S max burst", res);

	res = icsneoc2_settings_t1s_is_termination_enabled_for(device, netid, &termination);
	if(res == icsneoc2_error_success) {
		res = icsneoc2_settings_t1s_termination_for_set(device, netid, true);
		if(res != icsneoc2_error_success) return print_error_code("Failed to enable T1S termination", res);
	} else if(res != icsneoc2_error_get_settings_failure) {
		return print_error_code("Failed to query T1S termination support", res);
	}

	return 0;
}

static void print_mac(const char* label, const uint8_t* mac) {
	printf("%s %02x:%02x:%02x:%02x:%02x:%02x", label, mac[0], mac[1], mac[2], mac[3], mac[4], mac[5]);
}

static void print_payload_hex(const uint8_t* data, size_t length) {
	for(size_t i = 0; i < length; ++i) {
		printf("%02x", data[i]);
		if(i + 1 < length) {
			printf(" ");
		}
	}
	printf("\n");
}

static void build_loopback_frame(uint8_t* frame_data, size_t frame_size, const char* tx_name, const char* rx_name) {
	const char* tx_label = tx_name ? tx_name : "TX";
	const char* rx_label = rx_name ? rx_name : "RX";

	// Build one recognizable Ethernet frame so the receive side can match exactly what we sent.
	memset(frame_data, 0, frame_size);
	frame_data[0] = 0x00;
	frame_data[1] = 0xFC;
	frame_data[2] = 0x70;
	frame_data[3] = 0x00;
	frame_data[4] = 0x00;
	frame_data[5] = 0x02;
	frame_data[6] = 0x00;
	frame_data[7] = 0xFC;
	frame_data[8] = 0x70;
	frame_data[9] = 0x00;
	frame_data[10] = 0x00;
	frame_data[11] = 0x01;
	frame_data[12] = (uint8_t)((LOOPBACK_ETHER_TYPE >> 8) & 0xFF);
	frame_data[13] = (uint8_t)(LOOPBACK_ETHER_TYPE & 0xFF);
	snprintf((char*)&frame_data[14], frame_size - 14, "C2 T1S loopback %s->%s", tx_label, rx_label);
}

static int message_matches_loopback_frame(icsneoc2_message_t* message, const uint8_t* expected, size_t expected_length, bool* matches) {
	uint8_t data[1600] = {0};
	size_t data_length = sizeof(data);
	uint16_t ether_type = 0;
	icsneoc2_error_t res;

	if(!expected || !matches || expected_length < 14) {
		return print_error_code("Invalid loopback match output", icsneoc2_error_invalid_parameters);
	}

	*matches = false;

	res = icsneoc2_message_eth_ether_type_get(message, &ether_type);
	if(res != icsneoc2_error_success) {
		return print_error_code("Failed to read loopback EtherType", res);
	}
	if(ether_type != LOOPBACK_ETHER_TYPE) {
		return 0;
	}

	res = icsneoc2_message_data_get(message, data, &data_length);
	if(res != icsneoc2_error_success) {
		return print_error_code("Failed to read loopback frame data", res);
	}

	if(data_length < 14) {
		return 0;
	}

	if(data_length >= expected_length) {
		*matches = memcmp(data, expected, expected_length) == 0;
	} else {
		*matches = memcmp(data, expected, data_length) == 0;
	}
	return 0;
}

static int print_ethernet_message(icsneoc2_message_t* message, const char* direction_label) {
	icsneoc2_netid_t netid = 0;
	char netid_name[64] = {0};
	uint8_t dst_mac[6] = {0};
	uint8_t src_mac[6] = {0};
	uint16_t ether_type = 0;
	icsneoc2_message_eth_t1s_flags_t t1s_flags = 0;
	uint8_t node_id = 0;
	uint8_t burst_count = 0;
	uint8_t symbol_type = 0;
	uint8_t data[1600] = {0};
	size_t data_length = sizeof(data);
	icsneoc2_error_t res;

	res = icsneoc2_message_netid_get(message, &netid);
	if(res != icsneoc2_error_success) return print_error_code("Failed to get message netid", res);
	if(get_netid_name(netid, netid_name, sizeof(netid_name)) != 0) return 1;

	res = icsneoc2_message_data_get(message, data, &data_length);
	if(res != icsneoc2_error_success) return print_error_code("Failed to get Ethernet data", res);

	printf("%s on %s: %zu bytes\n", direction_label, netid_name, data_length);

	res = icsneoc2_message_eth_mac_get(message, dst_mac, src_mac);
	if(res == icsneoc2_error_success) {
		print_mac("  Dst", dst_mac);
		printf("  ");
		print_mac("Src", src_mac);
		printf("\n");
	}

	res = icsneoc2_message_eth_ether_type_get(message, &ether_type);
	if(res == icsneoc2_error_success) {
		printf("  EtherType: 0x%04x\n", ether_type);
	}

	res = icsneoc2_message_eth_t1s_props_get(message, &t1s_flags, &node_id, &burst_count, &symbol_type);
	if(res == icsneoc2_error_success && (t1s_flags != 0 || node_id != 0 || burst_count != 0 || symbol_type != 0)) {
		printf("  T1S: node=%u burst=%u symbol=%u", (unsigned)node_id, (unsigned)burst_count, (unsigned)symbol_type);
		if(t1s_flags & ICSNEOC2_MESSAGE_ETH_T1S_FLAGS_IS_T1S_SYMBOL) printf(" [SYMBOL]");
		if(t1s_flags & ICSNEOC2_MESSAGE_ETH_T1S_FLAGS_IS_T1S_BURST) printf(" [BURST]");
		if(t1s_flags & ICSNEOC2_MESSAGE_ETH_T1S_FLAGS_TX_COLLISION) printf(" [TX_COLLISION]");
		if(t1s_flags & ICSNEOC2_MESSAGE_ETH_T1S_FLAGS_IS_T1S_WAKE) printf(" [WAKE]");
		printf("\n");
	}

	printf("  Payload bytes: ");
	if(data_length > 14) {
		print_payload_hex(&data[14], data_length - 14);
	} else {
		printf("<none>\n");
	}

	return 0;
}

static int transmit_loopback_frame(icsneoc2_device_t* device, icsneoc2_netid_t tx_netid, const char* tx_name, const char* rx_name) {
	icsneoc2_message_t* message = NULL;
	icsneoc2_error_t res;
	uint8_t frame_data[LOOPBACK_FRAME_SIZE] = {0};

	build_loopback_frame(frame_data, sizeof(frame_data), tx_name, rx_name);

	res = icsneoc2_message_eth_create(&message);
	if(res != icsneoc2_error_success) return print_error_code("Failed to create Ethernet message", res);

	res = icsneoc2_message_netid_set(message, tx_netid);
	if(res != icsneoc2_error_success) {
		icsneoc2_message_free(message);
		return print_error_code("Failed to set Ethernet netid", res);
	}

	res = icsneoc2_message_data_set(message, frame_data, sizeof(frame_data));
	if(res != icsneoc2_error_success) {
		icsneoc2_message_free(message);
		return print_error_code("Failed to set Ethernet payload", res);
	}

	printf("Transmitting one T1S loopback frame on %s...\n", tx_name ? tx_name : "selected TX network");
	res = icsneoc2_device_message_transmit(device, message);
	icsneoc2_message_free(message);
	if(res != icsneoc2_error_success) return print_error_code("Failed to transmit loopback frame", res);

	return 0;
}

static int poll_for_loopback_messages(icsneoc2_device_t* device,
		icsneoc2_netid_t tx_netid,
		icsneoc2_netid_t rx_netid,
		const char* tx_name,
		const char* rx_name,
		const uint8_t* expected_frame,
		size_t expected_frame_length) {
	bool saw_tx_echo = false;
	bool saw_rx_frame = false;

	printf("Polling for TX echo on %s and RX frame on %s...\n", tx_name, rx_name);
	for(size_t attempt = 0; attempt < 60 && !(saw_tx_echo && saw_rx_frame); ++attempt) {
		icsneoc2_message_t* message = NULL;
		icsneoc2_error_t res = icsneoc2_device_message_get(device, &message, 100);
		if(res != icsneoc2_error_success) {
			return print_error_code("Failed while polling for loopback messages", res);
		}
		if(message == NULL) {
			continue;
		}

		bool is_ethernet = false;
		bool matches_loopback = false;
		res = icsneoc2_message_is_ethernet(message, &is_ethernet);
		if(res != icsneoc2_error_success || !is_ethernet) {
			icsneoc2_message_free(message);
			continue;
		}

		icsneoc2_netid_t netid = 0;
		res = icsneoc2_message_netid_get(message, &netid);
		if(res != icsneoc2_error_success) {
			icsneoc2_message_free(message);
			return print_error_code("Failed to get polled netid", res);
		}

		if(netid != tx_netid && netid != rx_netid) {
			icsneoc2_message_free(message);
			continue;
		}

		// Ignore unrelated traffic on the selected ports and only count the frame this example transmitted.
		if(message_matches_loopback_frame(message, expected_frame, expected_frame_length, &matches_loopback) != 0) {
			icsneoc2_message_free(message);
			return 1;
		}
		if(!matches_loopback) {
			icsneoc2_message_free(message);
			continue;
		}

		bool is_tx = false;
		res = icsneoc2_message_is_transmit(message, &is_tx);
		if(res != icsneoc2_error_success) {
			icsneoc2_message_free(message);
			return print_error_code("Failed to determine TX status", res);
		}

		if(netid == tx_netid && is_tx && !saw_tx_echo) {
			if(print_ethernet_message(message, "TX echo") != 0) {
				icsneoc2_message_free(message);
				return 1;
			}
			saw_tx_echo = true;
		} else if(netid == rx_netid && !saw_rx_frame) {
			if(print_ethernet_message(message, "RX frame") != 0) {
				icsneoc2_message_free(message);
				return 1;
			}
			if(is_tx) {
				printf("  Note: RX port message was also marked as transmit.\n");
			}
			saw_rx_frame = true;
		}

		icsneoc2_message_free(message);
	}

	if(!saw_tx_echo || !saw_rx_frame) {
		printf("Loopback incomplete: saw_tx_echo=%s, saw_rx_frame=%s\n",
			saw_tx_echo ? "true" : "false",
			saw_rx_frame ? "true" : "false");
		printf("Confirm %s is physically connected to %s and both ports are configured for 10BASE-T1S.\n", tx_name, rx_name);
		return 1;
	}

	printf("Loopback complete: TX echo on %s and RX frame on %s were both observed.\n", tx_name, rx_name);
	return 0;
}

TC10

Download example

/*
 * TC10 example.
 *
 * Sends TC10 wake/sleep requests, queries TC10 status, or lists connected
 * devices and the Automotive Ethernet networks they support. Loosely based
 * on examples/python/tc10/tc10.py. If --serial is omitted, the first
 * available device is used.
 */

#include <icsneo/icsneoc2.h>
#include <icsneo/icsneoc2messages.h>

#include <ctype.h>
#include <stdbool.h>
#include <stdio.h>
#include <stdlib.h>
#include <string.h>

typedef struct {
	const char* serial;        /* may be NULL */
	const char** networks;     /* points into argv */
	size_t networks_count;
	bool send_wake;
	bool send_sleep;
	bool status;
	bool list;
} args_t;

static int print_error_code(const char* message, icsneoc2_error_t error);
static void str_tolower(char* s);
static bool resolve_netid(const char* name, icsneoc2_netid_t* out);
static void print_usage(const char* prog);
static int parse_args(int argc, char** argv, args_t* out);
static icsneoc2_device_info_t* find_device(icsneoc2_device_info_t* list, const char* serial);
static int list_devices(icsneoc2_device_info_t* list);
static const char* tc10_wake_status_str(icsneoc2_tc10_wake_status_t s);
static const char* tc10_sleep_status_str(icsneoc2_tc10_sleep_status_t s);

int main(int argc, char** argv) {
	args_t args;
	if(parse_args(argc, argv, &args) != 0) {
		return 1;
	}

	icsneoc2_device_info_t* found_devices = NULL;
	icsneoc2_error_t res = icsneoc2_device_enumerate(0, &found_devices);
	if(res != icsneoc2_error_success) {
		return print_error_code("Failed to enumerate devices", res);
	}
	if(found_devices == NULL) {
		fprintf(stderr, "error: no devices found\n");
		return 1;
	}

	if(args.list) {
		int rc = list_devices(found_devices);
		icsneoc2_enumeration_free(found_devices);
		return rc;
	}

	icsneoc2_device_info_t* info = find_device(found_devices, args.serial);
	if(info == NULL) {
		fprintf(stderr, "error: unable to find device %s\n", args.serial ? args.serial : "(any)");
		icsneoc2_enumeration_free(found_devices);
		return 1;
	}

	char description[256] = {0};
	size_t description_len = sizeof(description);
	res = icsneoc2_device_info_description_get(info, description, &description_len);
	if(res != icsneoc2_error_success) {
		icsneoc2_enumeration_free(found_devices);
		return print_error_code("Failed to get device description", res);
	}

	icsneoc2_device_t* device = NULL;
	res = icsneoc2_device_create(info, &device);
	if(res != icsneoc2_error_success) {
		icsneoc2_enumeration_free(found_devices);
		return print_error_code("Failed to create device", res);
	}

	printf("Opening device %s\n", description);
	res = icsneoc2_device_open(device, icsneoc2_open_options_default);
	if(res != icsneoc2_error_success) {
		icsneoc2_device_free(device);
		icsneoc2_enumeration_free(found_devices);
		return print_error_code("Failed to open device", res);
	}

	bool supports_tc10 = false;
	res = icsneoc2_device_supports_tc10(device, &supports_tc10);
	if(res != icsneoc2_error_success || !supports_tc10) {
		fprintf(stderr, "error: device does not support TC10 (%s)\n", description);
		icsneoc2_device_close(device);
		icsneoc2_device_free(device);
		icsneoc2_enumeration_free(found_devices);
		return 1;
	}

	int rc = 0;
	for(size_t i = 0; i < args.networks_count; ++i) {
		const char* name = args.networks[i];
		icsneoc2_netid_t netid = icsneoc2_netid_invalid;
		if(!resolve_netid(name, &netid)) {
			fprintf(stderr, "error: unknown network '%s'\n", name);
			rc = 1;
			break;
		}

		if(args.send_wake) {
			printf("requesting TC10 wake on network %s\n", name);
			res = icsneoc2_device_tc10_wake_request(device, netid);
			if(res != icsneoc2_error_success) {
				rc = print_error_code("Failed to send TC10 wake", res);
				break;
			}
		} else if(args.send_sleep) {
			printf("requesting TC10 sleep on network %s\n", name);
			res = icsneoc2_device_tc10_sleep_request(device, netid);
			if(res != icsneoc2_error_success) {
				rc = print_error_code("Failed to send TC10 sleep", res);
				break;
			}
		} else { /* args.status */
			icsneoc2_tc10_sleep_status_t sleep_s = icsneoc2_tc10_sleep_status_no_sleep_received;
			icsneoc2_tc10_wake_status_t wake_s = icsneoc2_tc10_wake_status_no_wake_received;
			res = icsneoc2_device_tc10_status_get(device, netid, &sleep_s, &wake_s);
			if(res != icsneoc2_error_success) {
				rc = print_error_code("Failed to get TC10 status", res);
				break;
			}
			printf("TC10 status on network %s: wake=%s sleep=%s\n", name,
				tc10_wake_status_str(wake_s), tc10_sleep_status_str(sleep_s));
		}
	}

	printf("Closing device %s\n", description);
	icsneoc2_device_close(device);
	icsneoc2_device_free(device);
	icsneoc2_enumeration_free(found_devices);
	return rc;
}

static int print_error_code(const char* message, icsneoc2_error_t error) {
	char error_str[64];
	size_t error_str_len = sizeof(error_str);
	icsneoc2_error_t res = icsneoc2_error_code_get(error, error_str, &error_str_len);
	if(res != icsneoc2_error_success) {
		printf("%s: Failed to get string for error code %d with error code %d\n", message, error, res);
		return res;
	}
	printf("%s: \"%s\" (%u)\n", message, error_str, error);
	return (int)error;
}

static void str_tolower(char* s) {
	for(; *s; ++s) {
		*s = (char)tolower((unsigned char)*s);
	}
}

/*
 * Resolve a network name (e.g. "ETHERNET_01", "ae_01") to its icsneoc2_netid_t
 * by iterating over all known netids and comparing names case-insensitively.
 */
static bool resolve_netid(const char* name, icsneoc2_netid_t* out) {
	char want[64];
	strncpy(want, name, sizeof(want) - 1);
	want[sizeof(want) - 1] = '\0';
	str_tolower(want);

	for(uint16_t i = 0; i < icsneoc2_netid_maxsize; ++i) {
		char buf[64];
		size_t buf_len = sizeof(buf);
		if(icsneoc2_netid_name_get((icsneoc2_netid_t)i, buf, &buf_len) != icsneoc2_error_success) {
			continue;
		}
		str_tolower(buf);
		if(strcmp(buf, want) == 0) {
			*out = (icsneoc2_netid_t)i;
			return true;
		}
	}
	return false;
}

static void print_usage(const char* prog) {
	printf("Usage:\n");
	printf("  %s --list\n", prog);
	printf("  %s [--serial SERIAL] --networks NET1 [NET2 ...] (--send-wake | --send-sleep | --status)\n", prog);
	printf("\n");
	printf("  --list                List connected devices and the Automotive Ethernet networks they support.\n");
	printf("  --serial SERIAL       Serial number of the device. If omitted, the first available device is used.\n");
	printf("  --networks NET ...    One or more network names (e.g. ETHERNET_01 AE_01). Consumes args until the next flag.\n");
	printf("  --send-wake           Trigger TC10 wake on the selected networks.\n");
	printf("  --send-sleep          Trigger TC10 sleep on the selected networks.\n");
	printf("  --status              Query TC10 wake/sleep status on the selected networks.\n");
	printf("  -h, --help            Show this message.\n");
}

static int parse_args(int argc, char** argv, args_t* out) {
	memset(out, 0, sizeof(*out));
	for(int i = 1; i < argc; ++i) {
		const char* a = argv[i];
		if(strcmp(a, "-h") == 0 || strcmp(a, "--help") == 0) {
			print_usage(argv[0]);
			exit(0);
		} else if(strcmp(a, "--serial") == 0) {
			if(i + 1 >= argc) {
				fprintf(stderr, "error: --serial requires a value\n");
				return 1;
			}
			out->serial = argv[++i];
		} else if(strcmp(a, "--send-wake") == 0) {
			out->send_wake = true;
		} else if(strcmp(a, "--send-sleep") == 0) {
			out->send_sleep = true;
		} else if(strcmp(a, "--status") == 0) {
			out->status = true;
		} else if(strcmp(a, "--list") == 0) {
			out->list = true;
		} else if(strcmp(a, "--networks") == 0) {
			if(i + 1 >= argc || argv[i + 1][0] == '-') {
				fprintf(stderr, "error: --networks requires at least one network name\n");
				return 1;
			}
			out->networks = (const char**)&argv[i + 1];
			size_t count = 0;
			while(i + 1 < argc && argv[i + 1][0] != '-') {
				++count;
				++i;
			}
			out->networks_count = count;
		} else {
			fprintf(stderr, "error: unknown argument '%s'\n", a);
			print_usage(argv[0]);
			return 1;
		}
	}

	if(out->list) {
		if(out->send_wake || out->send_sleep || out->status || out->networks_count > 0 || out->serial) {
			fprintf(stderr, "error: --list cannot be combined with other options\n");
			print_usage(argv[0]);
			return 1;
		}
		return 0;
	}

	if(out->networks_count == 0) {
		fprintf(stderr, "error: --networks is required\n");
		print_usage(argv[0]);
		return 1;
	}
	int action_count = (out->send_wake ? 1 : 0) + (out->send_sleep ? 1 : 0) + (out->status ? 1 : 0);
	if(action_count != 1) {
		fprintf(stderr, "error: exactly one of --send-wake, --send-sleep, or --status is required\n");
		print_usage(argv[0]);
		return 1;
	}
	return 0;
}

static const char* tc10_wake_status_str(icsneoc2_tc10_wake_status_t s) {
	switch(s) {
		case icsneoc2_tc10_wake_status_no_wake_received: return "no_wake_received";
		case icsneoc2_tc10_wake_status_wake_received:    return "wake_received";
		default:                                         return "unknown";
	}
}

static const char* tc10_sleep_status_str(icsneoc2_tc10_sleep_status_t s) {
	switch(s) {
		case icsneoc2_tc10_sleep_status_no_sleep_received: return "no_sleep_received";
		case icsneoc2_tc10_sleep_status_sleep_received:    return "sleep_received";
		case icsneoc2_tc10_sleep_status_sleep_failed:      return "sleep_failed";
		case icsneoc2_tc10_sleep_status_sleep_aborted:     return "sleep_aborted";
		default:                                           return "unknown";
	}
}

/*
 * Find a device matching the provided serial, or the first available device if serial is NULL.
 * Returns NULL on failure (caller is responsible for the enumeration list).
 */
static icsneoc2_device_info_t* find_device(icsneoc2_device_info_t* list, const char* serial) {
	if(serial == NULL) {
		return list;
	}
	for(icsneoc2_device_info_t* cur = list; cur != NULL; cur = icsneoc2_device_info_next(cur)) {
		char dev_serial[64] = {0};
		size_t dev_serial_len = sizeof(dev_serial);
		if(icsneoc2_device_info_serial_get(cur, dev_serial, &dev_serial_len) != icsneoc2_error_success) {
			continue;
		}
		if(strcmp(dev_serial, serial) == 0) {
			return cur;
		}
	}
	return NULL;
}

/*
 * List connected devices and the Automotive Ethernet networks each one supports.
 */
static int list_devices(icsneoc2_device_info_t* list) {
	size_t index = 0;
	for(icsneoc2_device_info_t* cur = list; cur != NULL; cur = icsneoc2_device_info_next(cur)) {
		char serial[64] = {0};
		size_t serial_len = sizeof(serial);
		(void)icsneoc2_device_info_serial_get(cur, serial, &serial_len);

		char description[256] = {0};
		size_t description_len = sizeof(description);
		(void)icsneoc2_device_info_description_get(cur, description, &description_len);

		printf("[%zu] %s (%s)\n", index++, description, serial);

		icsneoc2_device_t* device = NULL;
		icsneoc2_error_t res = icsneoc2_device_create(cur, &device);
		if(res != icsneoc2_error_success) {
			print_error_code("    Failed to create device", res);
			continue;
		}

		icsneoc2_open_options_t options = icsneoc2_open_options_default;
		options &= ~ICSNEOC2_OPEN_OPTIONS_SYNC_RTC;
		options &= ~ICSNEOC2_OPEN_OPTIONS_GO_ONLINE;
		res = icsneoc2_device_open(device, options);
		if(res != icsneoc2_error_success) {
			print_error_code("    Failed to open device", res);
			icsneoc2_device_free(device);
			continue;
		}

		bool supports_tc10 = false;
		(void)icsneoc2_device_supports_tc10(device, &supports_tc10);
		printf("    TC10 supported: %s\n", supports_tc10 ? "yes" : "no");

		size_t count = 0;
		res = icsneoc2_device_supported_tx_networks_get(device, NULL, &count);
		if(res != icsneoc2_error_success || count == 0) {
			icsneoc2_device_close(device);
			icsneoc2_device_free(device);
			continue;
		}

		icsneoc2_netid_t* nets = (icsneoc2_netid_t*)calloc(count, sizeof(icsneoc2_netid_t));
		if(nets == NULL) {
			fprintf(stderr, "    error: out of memory\n");
			icsneoc2_device_close(device);
			icsneoc2_device_free(device);
			continue;
		}

		res = icsneoc2_device_supported_tx_networks_get(device, nets, &count);
		if(res != icsneoc2_error_success) {
			print_error_code("    Failed to get supported networks", res);
			free(nets);
			icsneoc2_device_close(device);
			icsneoc2_device_free(device);
			continue;
		}

		printf("    Automotive Ethernet networks:\n");
		bool any = false;
		for(size_t i = 0; i < count; ++i) {
			icsneoc2_network_type_t type = icsneoc2_network_type_invalid;
			if(icsneoc2_netid_network_type_get(nets[i], &type) != icsneoc2_error_success) {
				continue;
			}
			if(type != icsneoc2_network_type_automotive_ethernet) {
				continue;
			}
			char name[64] = {0};
			size_t name_len = sizeof(name);
			if(icsneoc2_netid_name_get(nets[i], name, &name_len) != icsneoc2_error_success) {
				continue;
			}
			printf("      %s\n", name);
			any = true;
		}
		if(!any) {
			printf("      (none)\n");
		}

		free(nets);
		icsneoc2_device_close(device);
		icsneoc2_device_free(device);
	}
	return 0;
}