/* * libdivecomputer * * Copyright (C) 2015 Jef Driesen * * This library is free software; you can redistribute it and/or * modify it under the terms of the GNU Lesser General Public * License as published by the Free Software Foundation; either * version 2.1 of the License, or (at your option) any later version. * * This library is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU * Lesser General Public License for more details. * * You should have received a copy of the GNU Lesser General Public * License along with this library; if not, write to the Free Software * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, * MA 02110-1301 USA */ #ifdef HAVE_CONFIG_H #include "config.h" #endif #include #include #ifdef HAVE_UNISTD_H #include #endif #ifdef HAVE_GETOPT_H #include #endif #include #include #include #include "dctool.h" #include "common.h" #include "utils.h" static dc_status_t doread (dc_context_t *context, dc_descriptor_t *descriptor, dc_transport_t transport, const char *devname, unsigned int address, dc_buffer_t *buffer) { dc_status_t rc = DC_STATUS_SUCCESS; dc_iostream_t *iostream = NULL; dc_device_t *device = NULL; // Open the I/O stream. message ("Opening the I/O stream (%s, %s).\n", dctool_transport_name (transport), devname ? devname : "null"); rc = dctool_iostream_open (&iostream, context, descriptor, transport, devname); if (rc != DC_STATUS_SUCCESS) { ERROR ("Error opening the I/O stream."); goto cleanup; } // Open the device. message ("Opening the device (%s %s).\n", dc_descriptor_get_vendor (descriptor), dc_descriptor_get_product (descriptor)); rc = dc_device_open (&device, context, descriptor, iostream); if (rc != DC_STATUS_SUCCESS) { ERROR ("Error opening the device."); goto cleanup; } // Register the event handler. message ("Registering the event handler.\n"); int events = DC_EVENT_WAITING | DC_EVENT_PROGRESS | DC_EVENT_DEVINFO | DC_EVENT_CLOCK | DC_EVENT_VENDOR; rc = dc_device_set_events (device, events, dctool_event_cb, NULL); if (rc != DC_STATUS_SUCCESS) { ERROR ("Error registering the event handler."); goto cleanup; } // Register the cancellation handler. message ("Registering the cancellation handler.\n"); rc = dc_device_set_cancel (device, dctool_cancel_cb, NULL); if (rc != DC_STATUS_SUCCESS) { ERROR ("Error registering the cancellation handler."); goto cleanup; } // Read data from the internal memory. message ("Reading data from the internal memory.\n"); rc = dc_device_read (device, address, dc_buffer_get_data (buffer), dc_buffer_get_size (buffer)); if (rc != DC_STATUS_SUCCESS) { ERROR ("Error reading from the internal memory."); goto cleanup; } cleanup: dc_device_close (device); dc_iostream_close (iostream); return rc; } static int dctool_read_run (int argc, char *argv[], dc_context_t *context, dc_descriptor_t *descriptor) { int exitcode = EXIT_SUCCESS; dc_status_t status = DC_STATUS_SUCCESS; dc_buffer_t *buffer = NULL; dc_transport_t transport = dctool_transport_default (descriptor); // Default option values. unsigned int help = 0; const char *filename = NULL; unsigned int address = 0, have_address = 0; unsigned int count = 0, have_count = 0; // Parse the command-line options. int opt = 0; const char *optstring = "ht:a:c:o:"; #ifdef HAVE_GETOPT_LONG struct option options[] = { {"help", no_argument, 0, 'h'}, {"transport", required_argument, 0, 't'}, {"address", required_argument, 0, 'a'}, {"count", required_argument, 0, 'c'}, {"output", required_argument, 0, 'o'}, {0, 0, 0, 0 } }; while ((opt = getopt_long (argc, argv, optstring, options, NULL)) != -1) { #else while ((opt = getopt (argc, argv, optstring)) != -1) { #endif switch (opt) { case 'h': help = 1; break; case 't': transport = dctool_transport_type (optarg); break; case 'a': address = strtoul (optarg, NULL, 0); have_address = 1; break; case 'c': count = strtoul (optarg, NULL, 0); have_count = 1; break; case 'o': filename = optarg; break; default: return EXIT_FAILURE; } } argc -= optind; argv += optind; // Show help message. if (help) { dctool_command_showhelp (&dctool_read); return EXIT_SUCCESS; } // Check the transport type. if (transport == DC_TRANSPORT_NONE) { message ("No valid transport type specified.\n"); exitcode = EXIT_FAILURE; goto cleanup; } // Check mandatory arguments. if (!have_address || !have_count) { message ("No memory address or byte count specified.\n"); exitcode = EXIT_FAILURE; goto cleanup; } // Allocate a memory buffer. buffer = dc_buffer_new (count); dc_buffer_resize (buffer, count); if (buffer == NULL) { message ("Failed to allocate a memory buffer.\n"); exitcode = EXIT_FAILURE; goto cleanup; } // Read data from the internal memory. status = doread (context, descriptor, transport, argv[0], address, buffer); if (status != DC_STATUS_SUCCESS) { message ("ERROR: %s\n", dctool_errmsg (status)); exitcode = EXIT_FAILURE; goto cleanup; } // Write the buffer to file. dctool_file_write (filename, buffer); cleanup: dc_buffer_free (buffer); return exitcode; } const dctool_command_t dctool_read = { dctool_read_run, DCTOOL_CONFIG_DESCRIPTOR, "read", "Read data from the internal memory", "Usage:\n" " dctool read [options] \n" "\n" "Options:\n" #ifdef HAVE_GETOPT_LONG " -h, --help Show help message\n" " -t, --transport Transport type\n" " -a, --address
Memory address\n" " -c, --count Number of bytes\n" " -o, --output Output filename\n" #else " -h Show help message\n" " -t Transport type\n" " -a
Memory address\n" " -c Number of bytes\n" " -o Output filename\n" #endif };