mirror of https://github.com/nirenjan/libx52.git
				
				
				
			
		
			
				
	
	
		
			542 lines
		
	
	
		
			15 KiB
		
	
	
	
		
			C
		
	
	
			
		
		
	
	
			542 lines
		
	
	
		
			15 KiB
		
	
	
	
		
			C
		
	
	
| /*
 | |
|  * Saitek X52 Pro MFD & LED driver
 | |
|  *
 | |
|  * Copyright (C) 2015 Nirenjan Krishnan (nirenjan@nirenjan.org)
 | |
|  *
 | |
|  * SPDX-License-Identifier: GPL-2.0-only WITH Classpath-exception-2.0
 | |
|  */
 | |
| 
 | |
| /**
 | |
| @page x52cli Command Line Interface to libx52
 | |
| 
 | |
| \htmlonly
 | |
| <b>x52cli</b> - Command line interface to libx52
 | |
| \endhtmlonly
 | |
| 
 | |
| # SYNOPSIS
 | |
| <tt>\b x52cli \a commands [\a command-options]</tt>
 | |
| 
 | |
| # DESCRIPTION
 | |
| 
 | |
| \b x52cli is a command line interface to the X52 library that allows you to set
 | |
| the LEDs and different parameters on the multifunction display (MFD).
 | |
| 
 | |
| Running \b x52cli without any arguments will display a brief help message.
 | |
| 
 | |
| # COMMANDS
 | |
| 
 | |
| Commands are not case sensitive.
 | |
| 
 | |
| - <tt>\b bri { \b mfd | \b led } < \a brightness >
 | |
|   </tt>\n \manonly \fR \endmanonly
 | |
|   Set the brightness of the \b MFD or <b>LED</b>s. \a brightness can be any
 | |
|   numeric value between 0 and 128. Higher values are accepted, but may not have
 | |
|   the desired effect.
 | |
| 
 | |
| - <tt>\b mfd < \a line > < \a text > </tt>\n \manonly \fR \endmanonly
 | |
|   Set the text on the MFD \a line. \a line can be \c 0, \c 1 or \c 2, and refers
 | |
|   to the first, second or third line of the multifunction display respectively.
 | |
|   \a text cannot have embedded NUL characters (0x00) and must correspond with
 | |
|   the character map fo the MFD. \a text should be quoted in order to preserve
 | |
|   embedded whitespace. To pass raw hex values, use \b printf(1) as shown in the
 | |
|   \ref x52cli_examples section. Note that \a text is limited to a length of 16
 | |
|   characters. While you can pass in longer strings, they will be silenty
 | |
|   truncated.
 | |
| 
 | |
| - <tt>\b led < \a led-id > < \a state > </tt>\n \manonly \fR \endmanonly
 | |
|   Set the LED \a led-id to \a state. See \ref x52cli_leds for a list of
 | |
|   supported values.
 | |
| 
 | |
| - <tt>\b blink { \b on | \b off } </tt>\n \manonly \fR \endmanonly
 | |
|   Turn the \b blink state \b on or \b off.
 | |
| 
 | |
| - <tt>\b shift { \b on | \b off } </tt>\n \manonly \fR \endmanonly
 | |
|   Turn the \b shift indicator in the MFD \b on or \b off.
 | |
| 
 | |
| - <tt>\b clock { \b local | \b gmt } { \b 12hr | \b 24hr }
 | |
|   { \b ddmmyy | \b mmddyy | \b yymmdd }</tt>\n \manonly \fR \endmanonly
 | |
|   Set the clock 1 display to the current \b local or \b GMT time and date.
 | |
|   Clock can be configured to display in either \b 12hr or \b 24hr mode. Date
 | |
|   can be displayed in one of the following formats: \b DD-MM-YY, \b MM-DD-YY,
 | |
|   or \b YY-MM-DD.
 | |
| 
 | |
| - <tt>\b offset { \b 2 | \b 3 } < \a offset-val >
 | |
|   { \b 12hr | \b 24hr }</tt>\n \manonly \fR \endmanonly
 | |
|   Set the offset for clock \b 2 or \b 3 and configure them to display in either
 | |
|   \b 12hr or \b 24hr mode. \a offset-val is in minutes east of \b UTC and can
 | |
|   range from \a -1440 to \a +1440.
 | |
| 
 | |
| - <tt>\b time < \a hour > < \a minute >
 | |
|   { \b 12hr | \b 24hr }</tt>\n \manonly \fR \endmanonly
 | |
|   Set the time for clock 1 to <em>hour:minute</em> and configure it to display
 | |
|   in \b 12hr or \b 24hr mode.
 | |
| 
 | |
| - <tt>\b date <\a dd > <\a mm > <\a yy>
 | |
|   { \b ddmmyy | \b mmddyy | \b yymmdd }</tt>\n \manonly \fR \endmanonly
 | |
|   Set the date on the MFD to the values represented by \a dd, \a mm and \a yy in
 | |
|   the requested format.
 | |
| 
 | |
| - <tt>\b raw < \a wIndex > < \a wValue ></tt>\n \manonly \fR \endmanonly
 | |
|   Send a raw vendor control request to the connected joystick.\n
 | |
|   \warning You should only use the raw command if you know what you are doing.
 | |
|   Sending an invalid control sequence can potentially damage or destroy your
 | |
|   device.
 | |
| 
 | |
| @section x52cli_leds LEDs
 | |
| This is the list of LED IDs and corresponding states supported by the X52 Pro.
 | |
| Note that the \b on state is only allowed for the \b fire and \b throttle LEDs,
 | |
| and they do not support the \b red, \b amber and \b green states. The remaining
 | |
| LEDs do not support the \b on state, but support all the other states.
 | |
| 
 | |
| \note The \b led command is only supported on the X52 Pro, not on the X52.
 | |
| 
 | |
| ## LED IDs
 | |
| 
 | |
| - \b fire
 | |
| - \b a
 | |
| - \b b
 | |
| - \b d
 | |
| - \b e
 | |
| - \b t1
 | |
| - \b t2
 | |
| - \b t3
 | |
| - \b pov
 | |
| - \b clutch
 | |
| - \b throttle
 | |
| 
 | |
| ## States
 | |
| 
 | |
| - \b off
 | |
| - \b on
 | |
| - \b red
 | |
| - \b amber
 | |
| - \b green
 | |
| 
 | |
| # LIMITATIONS
 | |
| 
 | |
| \b x52cli does not maintain any state between invocations. As a result the
 | |
| \b clock command will reset the relative offsets for clocks 2 and 3 back to 0
 | |
| and configure them to be in 12 hour mode. To work around this, use the \b date
 | |
| and \b time commands instead to manually configure the date and time.
 | |
| 
 | |
| # PERMISSIONS
 | |
| 
 | |
| You must have write permissions to the USB device in order to use the \b libx52
 | |
| library, and by extension, \b x52cli.
 | |
| 
 | |
| The simplest method to obtain such permissions is to run \b x52cli as root,
 | |
| possibly through \b sudo(8)
 | |
| 
 | |
| @section x52cli_examples EXAMPLES
 | |
| 
 | |
| - Turn off the T1 LED.
 | |
|   > <tt>\b x52cli led t1 off</tt>
 | |
| 
 | |
| - Turn the B LED to Amber.
 | |
|   > <tt>\b x52cli led B amber</tt>
 | |
| 
 | |
| - Set line 1 of the MFD to display "Hello World"
 | |
|   > <tt>\b x52cli mfd 0 "Hello World"</tt>
 | |
| 
 | |
| - Set line 2 of the MFD to display "¿Cómo Estás?"
 | |
|   > <tt>\b x52cli mfd 1 "$(printf '\\x9FC\\xE2mo Est\\xE0s?')"</tt>
 | |
| 
 | |
| */
 | |
| 
 | |
| #define _GNU_SOURCE
 | |
| #include "config.h"
 | |
| #include <stdio.h>
 | |
| #include <stdint.h>
 | |
| #include <stdlib.h>
 | |
| #include <string.h>
 | |
| #include <unistd.h>
 | |
| #include <time.h>
 | |
| 
 | |
| #include "libx52.h"
 | |
| 
 | |
| struct string_map {
 | |
|     const char *key;
 | |
|     union {
 | |
|         libx52_clock_id     clock_id;
 | |
|         libx52_clock_format clock_format;
 | |
|         libx52_date_format  date_format;
 | |
|         libx52_led_id       led_id;
 | |
|         libx52_led_state    led_state;
 | |
|         int                 int_val;
 | |
|         intptr_t            ptr_val;
 | |
|     } value;
 | |
| };
 | |
| 
 | |
| // Max 4 arguments for now
 | |
| #define MAX_ARGS    4
 | |
| 
 | |
| typedef int (*handler_cb)(libx52_device *x52, void *args[]);
 | |
| struct command_handler {
 | |
|     handler_cb handler;
 | |
|     int num_args;
 | |
|     const struct string_map *maps[MAX_ARGS];
 | |
|     const char *help;
 | |
| };
 | |
| 
 | |
| #define SAVE_ARGUMENT(type, name, arg) type name = (type)(uintptr_t)(arg)
 | |
| #define PARSE_ARGUMENT(type, arg) ((type)(uintptr_t)(arg))
 | |
| 
 | |
| #define MAP(name)   name##_map
 | |
| #define DEFINE_MAP(name) static const struct string_map MAP(name)[]
 | |
| #define MAP_CLOCK_ID(id)        {.key = #id, .value.clock_id = LIBX52_CLOCK_ ## id}
 | |
| #define MAP_CLOCK_FORMAT(fmt)   {.key = #fmt, .value.clock_format = LIBX52_CLOCK_FORMAT_ ## fmt}
 | |
| #define MAP_DATE_FORMAT(fmt)    {.key = #fmt, .value.date_format = LIBX52_DATE_FORMAT_ ## fmt}
 | |
| #define MAP_LED_ID(id)          {.key = #id, .value.led_id = LIBX52_LED_ ## id}
 | |
| #define MAP_LED_STATE(state)    {.key = #state, .value.led_state = LIBX52_LED_STATE_ ## state}
 | |
| #define MAP_INT(str, val)       {.key = str, .value.int_val = val}
 | |
| #define MAP_TERMINATOR  MAP_INT(NULL, -1)
 | |
| 
 | |
| /**
 | |
|  * Parse a string and match it with a corresponding value
 | |
|  * Maps are arrays which contain a terminating entry with a NULL key.
 | |
|  *
 | |
|  * Return 0 if no match
 | |
|  */
 | |
| static int map_lookup(const struct string_map *map, const char *str, struct string_map *result)
 | |
| {
 | |
|     const struct string_map *map_index = map;
 | |
| 
 | |
|     /* Search through the map for a matching key */
 | |
|     while (map_index->key) {
 | |
|         if (!strcasecmp(str, map_index->key)) {
 | |
|             result->value = map_index->value;
 | |
|             return 1;
 | |
|         }
 | |
| 
 | |
|         map_index++;
 | |
|     }
 | |
| 
 | |
|     return 0;
 | |
| }
 | |
| 
 | |
| /* Map for LED state */
 | |
| DEFINE_MAP(led_state) = {
 | |
|     MAP_LED_STATE(OFF),
 | |
|     MAP_LED_STATE(ON),
 | |
|     MAP_LED_STATE(RED),
 | |
|     MAP_LED_STATE(AMBER),
 | |
|     MAP_LED_STATE(GREEN),
 | |
|     MAP_TERMINATOR
 | |
| };
 | |
| 
 | |
| /* Map for LED identifier */
 | |
| DEFINE_MAP(led_id) = {
 | |
|     MAP_LED_ID(FIRE),
 | |
|     MAP_LED_ID(A),
 | |
|     MAP_LED_ID(B),
 | |
|     MAP_LED_ID(D),
 | |
|     MAP_LED_ID(E),
 | |
|     MAP_LED_ID(T1),
 | |
|     MAP_LED_ID(T2),
 | |
|     MAP_LED_ID(T3),
 | |
|     MAP_LED_ID(POV),
 | |
|     MAP_LED_ID(CLUTCH),
 | |
|     MAP_LED_ID(THROTTLE),
 | |
|     MAP_TERMINATOR
 | |
| };
 | |
| 
 | |
| /* Map for date format */
 | |
| DEFINE_MAP(date_format) = {
 | |
|     MAP_DATE_FORMAT(DDMMYY),
 | |
|     MAP_DATE_FORMAT(MMDDYY),
 | |
|     MAP_DATE_FORMAT(YYMMDD),
 | |
|     MAP_TERMINATOR
 | |
| };
 | |
| 
 | |
| /* Map for brightness setting */
 | |
| DEFINE_MAP(brightness_targets) = {
 | |
|     MAP_INT( "mfd", 1 ),
 | |
|     MAP_INT( "led", 0 ),
 | |
|     MAP_TERMINATOR
 | |
| };
 | |
| 
 | |
| /* Map for blink/shift on/off */
 | |
| DEFINE_MAP(on_off) = {
 | |
|     MAP_INT( "off", 0  ),
 | |
|     MAP_INT( "on",  1  ),
 | |
|     MAP_TERMINATOR
 | |
| };
 | |
| 
 | |
| /* Map for clock 0 timezone */
 | |
| DEFINE_MAP(clock0_timezone) = {
 | |
|     MAP_INT( "gmt",     0  ),
 | |
|     MAP_INT( "local",   1  ),
 | |
|     MAP_TERMINATOR
 | |
| };
 | |
| 
 | |
| /* Map for identifying the clock for the timezone */
 | |
| DEFINE_MAP(clocks) = {
 | |
|     MAP_CLOCK_ID(1),
 | |
|     MAP_CLOCK_ID(2),
 | |
|     MAP_CLOCK_ID(3),
 | |
|     MAP_TERMINATOR
 | |
| };
 | |
| 
 | |
| /* Map for identifying the time format */
 | |
| DEFINE_MAP(time_format) = {
 | |
|     MAP_CLOCK_FORMAT(12HR),
 | |
|     MAP_CLOCK_FORMAT(24HR),
 | |
|     MAP_TERMINATOR
 | |
| };
 | |
| 
 | |
| static int update_led(libx52_device *x52, void *args[])
 | |
| {
 | |
|     return libx52_set_led_state(x52,
 | |
|         PARSE_ARGUMENT(libx52_led_id, args[0]),
 | |
|         PARSE_ARGUMENT(libx52_led_state, args[1]));
 | |
| }
 | |
| 
 | |
| static int update_bri(libx52_device *x52, void *args[])
 | |
| {
 | |
|     unsigned long int brightness = strtoul(args[1], NULL, 0);
 | |
| 
 | |
|     return libx52_set_brightness(x52,
 | |
|         PARSE_ARGUMENT(uint8_t, args[0]), (uint16_t)brightness);
 | |
| }
 | |
| 
 | |
| static int update_mfd(libx52_device *x52, void *args[])
 | |
| {
 | |
|     uint8_t line = (uint8_t)strtoul(args[0], NULL, 0);
 | |
|     uint8_t length = strlen(args[1]);
 | |
| 
 | |
|     return libx52_set_text(x52, line, args[1], length);
 | |
| }
 | |
| 
 | |
| static int update_blink(libx52_device *x52, void *args[])
 | |
| {
 | |
|     return libx52_set_blink(x52, PARSE_ARGUMENT(int, args[0]));
 | |
| }
 | |
| 
 | |
| static int update_shift(libx52_device *x52, void *args[])
 | |
| {
 | |
|     return libx52_set_shift(x52, PARSE_ARGUMENT(int, args[0]));
 | |
| }
 | |
| 
 | |
| static int update_clock(libx52_device *x52, void *args[])
 | |
| {
 | |
|     int rc;
 | |
|     rc = libx52_set_clock(x52, time(NULL),
 | |
|         PARSE_ARGUMENT(int, args[0]));
 | |
| 
 | |
|     if (!rc) {
 | |
|         rc = libx52_set_clock_format(x52, LIBX52_CLOCK_1,
 | |
|                 PARSE_ARGUMENT(libx52_clock_format, args[1]));
 | |
|     }
 | |
| 
 | |
|     if (!rc) {
 | |
|         rc = libx52_set_date_format(x52,
 | |
|             PARSE_ARGUMENT(libx52_date_format, args[2]));
 | |
|     }
 | |
| 
 | |
|     return rc;
 | |
| }
 | |
| 
 | |
| static int update_offset(libx52_device *x52, void *args[])
 | |
| {
 | |
|     int offset = (int)strtol(args[1], NULL, 0);
 | |
|     int rc;
 | |
|     SAVE_ARGUMENT(libx52_clock_id, clock, args[0]);
 | |
| 
 | |
|     rc = libx52_set_clock_timezone(x52, clock, offset);
 | |
| 
 | |
|     if (!rc) {
 | |
|         rc = libx52_set_clock_format(x52, clock,
 | |
|             PARSE_ARGUMENT(libx52_clock_format, args[2]));
 | |
|     }
 | |
| 
 | |
|     return rc;
 | |
| }
 | |
| 
 | |
| static int update_time(libx52_device *x52, void *args[])
 | |
| {
 | |
|     int hh = (int)strtol(args[0], NULL, 0);
 | |
|     int mm = (int)strtol(args[1], NULL, 0);
 | |
|     int rc;
 | |
| 
 | |
|     /* Set the time value */
 | |
|     rc = libx52_set_time(x52, hh, mm);
 | |
|     if (!rc) {
 | |
|         rc = libx52_set_clock_format(x52, LIBX52_CLOCK_1,
 | |
|             PARSE_ARGUMENT(libx52_clock_format, args[2]));
 | |
|     }
 | |
| 
 | |
|     return rc;
 | |
| }
 | |
| 
 | |
| static int update_date(libx52_device *x52, void *args[])
 | |
| {
 | |
|     int dd = (int)strtol(args[0], NULL, 0);
 | |
|     int mm = (int)strtol(args[1], NULL, 0);
 | |
|     int yy = (int)strtol(args[2], NULL, 0);
 | |
|     int rc;
 | |
| 
 | |
|     /* Set the date value */
 | |
|     rc = libx52_set_date(x52, dd, mm, yy);
 | |
|     if (!rc) {
 | |
|         rc = libx52_set_date_format(x52,
 | |
|             PARSE_ARGUMENT(libx52_date_format, args[3]));
 | |
|     }
 | |
| 
 | |
|     return rc;
 | |
| }
 | |
| 
 | |
| static int update_raw(libx52_device *x52, void *args[])
 | |
| {
 | |
|     uint16_t wIndex = (uint16_t)strtoul(args[0], NULL, 0);
 | |
|     uint16_t wValue = (uint16_t)strtoul(args[1], NULL, 0);
 | |
| 
 | |
|     return libx52_vendor_command(x52, wIndex, wValue);
 | |
| }
 | |
| 
 | |
| /* Commands for CLI */
 | |
| #define COMMANDS \
 | |
|     X(led,      LED_STATE,  "<led-id> <state>",     2, \
 | |
|         MAP(led_id), MAP(led_state)) \
 | |
|     X(bri,      BRIGHTNESS, "{mfd | led} <brightness level>", 2, \
 | |
|         MAP(brightness_targets), NULL) \
 | |
|     X(mfd,      MFD_TEXT,   "<line> <text in quotes>", 2, \
 | |
|         NULL, NULL) \
 | |
|     X(blink,    BLINK,      "{ on | off }", 1, \
 | |
|         MAP(on_off)) \
 | |
|     X(shift,    SHIFT,      "{ on | off }", 1, \
 | |
|         MAP(on_off)) \
 | |
|     X(clock,    CLOCK, \
 | |
|         "{local | gmt} {12hr | 24hr} {ddmmyy | mmddyy | yymmdd}", \
 | |
|         3, MAP(clock0_timezone), MAP(time_format), MAP(date_format)) \
 | |
|     X(offset,   OFFSET, \
 | |
|         "{2 | 3} <offset from clock 1 in minutes> {12hr | 24hr}", \
 | |
|         3, MAP(clocks), NULL, MAP(time_format)) \
 | |
|     X(time,     TIME, \
 | |
|         "<hour> <minute> {12hr | 24hr}", 3, \
 | |
|         NULL, NULL, MAP(time_format)) \
 | |
|     X(date,     DATE, \
 | |
|         "<dd> <mm> <yy> {ddmmyy | mmddyy | yymmdd}", 4, \
 | |
|         NULL, NULL, NULL, MAP(date_format)) \
 | |
|     X(raw,      RAW,        "<wIndex> <wValue>", 2, NULL, NULL)
 | |
| 
 | |
| /* Enums for command identification */
 | |
| #define X(cmd, en, help, args, ...) X52_CTL_CMD_ ## en,
 | |
| enum {
 | |
|     COMMANDS
 | |
| 
 | |
|     X52_CTL_CMD_MAX
 | |
| };
 | |
| #undef X
 | |
| 
 | |
| /* Map for commands */
 | |
| #define X(cmd, en, help, args, ...) MAP_INT( #cmd, X52_CTL_CMD_ ## en),
 | |
| DEFINE_MAP(command) = {
 | |
|     COMMANDS
 | |
| 
 | |
|     MAP_TERMINATOR
 | |
| };
 | |
| #undef X
 | |
| 
 | |
| /* Command handlers */
 | |
| #define X(cmd, en, help, args, ...) \
 | |
|     [X52_CTL_CMD_ ## en] = { \
 | |
|         update_ ## cmd, \
 | |
|         args, \
 | |
|         { __VA_ARGS__ }, \
 | |
|         #cmd " " help \
 | |
|     },
 | |
| const struct command_handler handlers[X52_CTL_CMD_MAX] = {
 | |
|     COMMANDS
 | |
| };
 | |
| #undef X
 | |
| 
 | |
| static void do_help(const struct command_handler *cmd)
 | |
| {
 | |
|     int i;
 | |
|     if (cmd) {
 | |
|         fprintf(stderr, "Command usage: %s\n", cmd->help);
 | |
|     } else {
 | |
|         printf("\nCommands:\n");
 | |
|         for (i = 0; i < X52_CTL_CMD_MAX; i++) {
 | |
|             printf("\t%s\n", handlers[i].help);
 | |
|         }
 | |
| 
 | |
|         printf("\nWARNING: raw command may damage your device\n\n");
 | |
|     }
 | |
| }
 | |
| 
 | |
| int main(int argc, char **argv)
 | |
| {
 | |
|     libx52_device *x52;
 | |
|     struct string_map result;
 | |
|     const struct command_handler *cmd;
 | |
|     int i;
 | |
|     void *args[MAX_ARGS];
 | |
|     int rc;
 | |
| 
 | |
|     if (argc < 2) {
 | |
|         fprintf(stderr, "Usage: %s <command> [arguments]\n", argv[0]);
 | |
|         do_help(NULL);
 | |
|         return 1;
 | |
|     }
 | |
| 
 | |
|     if (!map_lookup(command_map, argv[1], &result)) {
 | |
|         fprintf(stderr, "Unsupported command %s\n", argv[1]);
 | |
|         do_help(NULL);
 | |
|         return 1;
 | |
|     }
 | |
| 
 | |
|     cmd = &handlers[result.value.int_val];
 | |
|     if (!cmd->handler) {
 | |
|         fprintf(stderr, "Command %s not implemented yet!\n", argv[1]);
 | |
|         return 1;
 | |
|     }
 | |
| 
 | |
|     if (cmd->num_args > argc - 2) {
 | |
|         fprintf(stderr, "Insufficient arguments for command %s\n", argv[1]);
 | |
|         do_help(cmd);
 | |
|         return 1;
 | |
|     }
 | |
| 
 | |
|     /* Clear the arguments array */
 | |
|     memset(args, 0, sizeof(args));
 | |
| 
 | |
|     for (i = 0; i < cmd->num_args; i++) {
 | |
|         if (cmd->maps[i]) {
 | |
|             if (!map_lookup(cmd->maps[i], argv[2+i], &result)) {
 | |
|                 fprintf(stderr, "Invalid argument %s\n", argv[2+i]);
 | |
|                 return 1;
 | |
|             }
 | |
|             args[i] = (void *)result.value.ptr_val;
 | |
|         } else {
 | |
|             args[i] = argv[2+i];
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     /* Initialize libx52 */
 | |
|     rc = libx52_init(&x52);
 | |
| 
 | |
|     if (rc != LIBX52_SUCCESS) {
 | |
|         fprintf(stderr, "Error initializing X52 library: %s\n", libx52_strerror(rc));
 | |
|         return 1;
 | |
|     }
 | |
| 
 | |
|     /* Make sure we are connected to the joystick */
 | |
|     rc = libx52_connect(x52);
 | |
|     if (rc != LIBX52_SUCCESS) {
 | |
|         fprintf(stderr, "Error connecting to joystick: %s\n", libx52_strerror(rc));
 | |
|         return 1;
 | |
|     }
 | |
| 
 | |
|     rc = (*(cmd->handler))(x52, args);
 | |
|     if (rc != LIBX52_SUCCESS) {
 | |
|         fprintf(stderr, "Error: %s\n", libx52_strerror(rc));
 | |
|     }
 | |
| 
 | |
|     libx52_update(x52);
 | |
| 
 | |
|     libx52_exit(x52);
 | |
| 
 | |
|     return rc;
 | |
| }
 |