Home | History | Annotate | Download | only in src
      1 /**
      2  * \file util.c
      3  *
      4  * This file contains generic utility functions such as can be
      5  * used for debugging for example.
      6  *
      7  * Copyright (C) 2005-2007 Linus Walleij <triad (at) df.lth.se>
      8  *
      9  * This library is free software; you can redistribute it and/or
     10  * modify it under the terms of the GNU Lesser General Public
     11  * License as published by the Free Software Foundation; either
     12  * version 2 of the License, or (at your option) any later version.
     13  *
     14  * This library is distributed in the hope that it will be useful,
     15  * but WITHOUT ANY WARRANTY; without even the implied warranty of
     16  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
     17  * Lesser General Public License for more details.
     18  *
     19  * You should have received a copy of the GNU Lesser General Public
     20  * License along with this library; if not, write to the
     21  * Free Software Foundation, Inc., 59 Temple Place - Suite 330,
     22  * Boston, MA 02111-1307, USA.
     23  */
     24 
     25 /* MSVC does not have these */
     26 #ifndef _MSC_VER
     27 #include <sys/time.h>
     28 #include <unistd.h>
     29 #endif
     30 
     31 #include <stdio.h>
     32 #include <stdlib.h>
     33 #include <errno.h>
     34 #include <sys/stat.h>
     35 #include <fcntl.h>
     36 #include <string.h>
     37 #include "config.h"
     38 #include "libmtp.h"
     39 #include "util.h"
     40 
     41 
     42 /**
     43  * This prints to stdout info about device being UNKNOWN, its
     44  * ids, and libmtp's version number.
     45  *
     46  * @param dev_number the device number
     47  * @param id_vendor vendor ID from the usb_device_desc struct
     48  * @param id_product product ID from the usb_device_desc struct
     49  */
     50 void device_unknown(const int dev_number, const int id_vendor, const int id_product)
     51 {
     52   // This device is unknown to the developers
     53   LIBMTP_ERROR("Device %d (VID=%04x and PID=%04x) is UNKNOWN in libmtp v%s.\n",
     54     dev_number,
     55     id_vendor,
     56     id_product,
     57     LIBMTP_VERSION_STRING);
     58   LIBMTP_ERROR("Please report this VID/PID and the device model to the "
     59                "libmtp development team\n");
     60   /*
     61    * Trying to get iManufacturer or iProduct from the device at this
     62    * point would require opening a device handle, that we don't want
     63    * to do right now. (Takes time for no good enough reason.)
     64    */
     65 }
     66 
     67 /**
     68  * This dumps out a number of bytes to a textual, hexadecimal
     69  * dump.
     70  *
     71  * @param f the file to dump to (e.g. stdout or stderr)
     72  * @param buf a pointer to the buffer containing the bytes to
     73  *            be dumped out in hex
     74  * @param n the number of bytes to dump from this buffer
     75  */
     76 void data_dump (FILE *f, void *buf, uint32_t n)
     77 {
     78   unsigned char *bp = (unsigned char *) buf;
     79   uint32_t i;
     80 
     81   for (i = 0; i < n; i++) {
     82     fprintf(f, "%02x ", *bp);
     83     bp++;
     84   }
     85   fprintf(f, "\n");
     86 }
     87 
     88 /**
     89  * This dumps out a number of bytes to a textual, hexadecimal
     90  * dump, and also prints out the string ASCII representation
     91  * for each line of bytes. It will also print the memory address
     92  * offset from a certain boundry.
     93  *
     94  * @param f the file to dump to (e.g. stdout or stderr)
     95  * @param buf a pointer to the buffer containing the bytes to
     96  *            be dumped out in hex
     97  * @param n the number of bytes to dump from this buffer
     98  * @param dump_boundry the address offset to start at (usually 0)
     99  */
    100 void data_dump_ascii (FILE *f, void *buf, uint32_t n, uint32_t dump_boundry)
    101 {
    102   uint32_t remain = n;
    103   uint32_t ln, lc;
    104   int i;
    105   unsigned char *bp = (unsigned char *) buf;
    106 
    107   lc = 0;
    108   while (remain) {
    109     fprintf(f, "\t%04x:", dump_boundry-0x10);
    110 
    111     ln = ( remain > 16 ) ? 16 : remain;
    112 
    113     for (i = 0; i < ln; i++) {
    114       if ( ! (i%2) ) fprintf(f, " ");
    115       fprintf(f, "%02x", bp[16*lc+i]);
    116     }
    117 
    118     if ( ln < 16 ) {
    119       int width = ((16-ln)/2)*5 + (2*(ln%2));
    120       fprintf(f, "%*.*s", width, width, "");
    121     }
    122 
    123     fprintf(f, "\t");
    124     for (i = 0; i < ln; i++) {
    125       unsigned char ch= bp[16*lc+i];
    126       fprintf(f, "%c", ( ch >= 0x20 && ch <= 0x7e ) ?
    127 	      ch : '.');
    128     }
    129     fprintf(f, "\n");
    130 
    131     lc++;
    132     remain -= ln;
    133     dump_boundry += ln;
    134   }
    135 }
    136 
    137 #ifndef HAVE_STRNDUP
    138 char *strndup (const char *s, size_t n)
    139 {
    140   size_t len = strlen (s);
    141   char *ret;
    142 
    143   if (len <= n)
    144     return strdup (s);
    145 
    146   ret = malloc(n + 1);
    147   strncpy(ret, s, n);
    148   ret[n] = '\0';
    149   return ret;
    150 }
    151 #endif
    152 
    153