Chris@40: /* Chris@40: ** Copyright (C) 1999-2014 Erik de Castro Lopo Chris@40: ** Chris@40: ** All rights reserved. Chris@40: ** Chris@40: ** Redistribution and use in source and binary forms, with or without Chris@40: ** modification, are permitted provided that the following conditions are Chris@40: ** met: Chris@40: ** Chris@40: ** * Redistributions of source code must retain the above copyright Chris@40: ** notice, this list of conditions and the following disclaimer. Chris@40: ** * Redistributions in binary form must reproduce the above copyright Chris@40: ** notice, this list of conditions and the following disclaimer in Chris@40: ** the documentation and/or other materials provided with the Chris@40: ** distribution. Chris@40: ** * Neither the author nor the names of any contributors may be used Chris@40: ** to endorse or promote products derived from this software without Chris@40: ** specific prior written permission. Chris@40: ** Chris@40: ** THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS Chris@40: ** "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED Chris@40: ** TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR Chris@40: ** PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR Chris@40: ** CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, Chris@40: ** EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, Chris@40: ** PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; Chris@40: ** OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, Chris@40: ** WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR Chris@40: ** OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF Chris@40: ** ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. Chris@40: */ Chris@40: Chris@40: #include Chris@40: #include Chris@40: #include Chris@40: #include Chris@40: #include Chris@40: #include Chris@40: Chris@40: #include Chris@40: Chris@40: #include "common.h" Chris@40: Chris@40: #define BUFFER_LEN (1 << 16) Chris@40: Chris@40: #if (defined (WIN32) || defined (_WIN32)) Chris@40: #include Chris@40: #endif Chris@40: Chris@40: static void usage_exit (const char *progname) ; Chris@40: Chris@40: static void info_dump (const char *filename) ; Chris@40: static int instrument_dump (const char *filename) ; Chris@40: static int broadcast_dump (const char *filename) ; Chris@40: static int chanmap_dump (const char *filename) ; Chris@40: static int cart_dump (const char *filename) ; Chris@40: static void total_dump (void) ; Chris@40: Chris@40: static double total_seconds = 0.0 ; Chris@40: Chris@40: int Chris@40: main (int argc, char *argv []) Chris@40: { int k ; Chris@40: Chris@40: if (argc < 2 || strcmp (argv [1], "--help") == 0 || strcmp (argv [1], "-h") == 0) Chris@40: usage_exit (program_name (argv [0])) ; Chris@40: Chris@40: if (strcmp (argv [1], "--instrument") == 0) Chris@40: { int error = 0 ; Chris@40: Chris@40: for (k = 2 ; k < argc ; k++) Chris@40: error += instrument_dump (argv [k]) ; Chris@40: return error ; Chris@40: } ; Chris@40: Chris@40: if (strcmp (argv [1], "--broadcast") == 0) Chris@40: { int error = 0 ; Chris@40: Chris@40: for (k = 2 ; k < argc ; k++) Chris@40: error += broadcast_dump (argv [k]) ; Chris@40: return error ; Chris@40: } ; Chris@40: Chris@40: if (strcmp (argv [1], "--channel-map") == 0) Chris@40: { int error = 0 ; Chris@40: Chris@40: for (k = 2 ; k < argc ; k++) Chris@40: error += chanmap_dump (argv [k]) ; Chris@40: return error ; Chris@40: } ; Chris@40: Chris@40: if (strcmp (argv [1], "--cart") == 0) Chris@40: { int error = 0 ; Chris@40: Chris@40: for (k = 2 ; k < argc ; k++) Chris@40: error += cart_dump (argv [k]) ; Chris@40: return error ; Chris@40: } ; Chris@40: Chris@40: for (k = 1 ; k < argc ; k++) Chris@40: info_dump (argv [k]) ; Chris@40: Chris@40: if (argc > 2) Chris@40: total_dump () ; Chris@40: Chris@40: return 0 ; Chris@40: } /* main */ Chris@40: Chris@40: /*============================================================================== Chris@40: ** Print version and usage. Chris@40: */ Chris@40: Chris@40: static double data [BUFFER_LEN] ; Chris@40: Chris@40: static void Chris@40: usage_exit (const char *progname) Chris@40: { printf ("Usage :\n %s ...\n", progname) ; Chris@40: printf (" Prints out information about one or more sound files.\n\n") ; Chris@40: printf (" %s --instrument \n", progname) ; Chris@40: printf (" Prints out the instrument data for the given file.\n\n") ; Chris@40: printf (" %s --broadcast \n", progname) ; Chris@40: printf (" Prints out the broadcast WAV info for the given file.\n\n") ; Chris@40: printf (" %s --channel-map \n", progname) ; Chris@40: printf (" Prints out the channel map for the given file.\n\n") ; Chris@40: printf (" %s --cart \n", progname) ; Chris@40: printf (" Prints out the cart chunk WAV info for the given file.\n\n") ; Chris@40: Chris@40: printf ("Using %s.\n\n", sf_version_string ()) ; Chris@40: #if (defined (_WIN32) || defined (WIN32)) Chris@40: printf ("This is a Unix style command line application which\n" Chris@40: "should be run in a MSDOS box or Command Shell window.\n\n") ; Chris@40: printf ("Sleeping for 5 seconds before exiting.\n\n") ; Chris@40: fflush (stdout) ; Chris@40: Chris@40: Sleep (5 * 1000) ; Chris@40: #endif Chris@40: exit (1) ; Chris@40: } /* usage_exit */ Chris@40: Chris@40: /*============================================================================== Chris@40: ** Dumping of sndfile info. Chris@40: */ Chris@40: Chris@40: static double data [BUFFER_LEN] ; Chris@40: Chris@40: static double Chris@40: calc_decibels (SF_INFO * sfinfo, double max) Chris@40: { double decibels ; Chris@40: Chris@40: switch (sfinfo->format & SF_FORMAT_SUBMASK) Chris@40: { case SF_FORMAT_PCM_U8 : Chris@40: case SF_FORMAT_PCM_S8 : Chris@40: decibels = max / 0x80 ; Chris@40: break ; Chris@40: Chris@40: case SF_FORMAT_PCM_16 : Chris@40: decibels = max / 0x8000 ; Chris@40: break ; Chris@40: Chris@40: case SF_FORMAT_PCM_24 : Chris@40: decibels = max / 0x800000 ; Chris@40: break ; Chris@40: Chris@40: case SF_FORMAT_PCM_32 : Chris@40: decibels = max / 0x80000000 ; Chris@40: break ; Chris@40: Chris@40: case SF_FORMAT_FLOAT : Chris@40: case SF_FORMAT_DOUBLE : Chris@40: decibels = max / 1.0 ; Chris@40: break ; Chris@40: Chris@40: default : Chris@40: decibels = max / 0x8000 ; Chris@40: break ; Chris@40: } ; Chris@40: Chris@40: return 20.0 * log10 (decibels) ; Chris@40: } /* calc_decibels */ Chris@40: Chris@40: static const char * Chris@40: format_duration_str (double seconds) Chris@40: { static char str [128] ; Chris@40: int hrs, min ; Chris@40: double sec ; Chris@40: Chris@40: memset (str, 0, sizeof (str)) ; Chris@40: Chris@40: hrs = (int) (seconds / 3600.0) ; Chris@40: min = (int) ((seconds - (hrs * 3600.0)) / 60.0) ; Chris@40: sec = seconds - (hrs * 3600.0) - (min * 60.0) ; Chris@40: Chris@40: snprintf (str, sizeof (str) - 1, "%02d:%02d:%06.3f", hrs, min, sec) ; Chris@40: Chris@40: return str ; Chris@40: } /* format_duration_str */ Chris@40: Chris@40: static const char * Chris@40: generate_duration_str (SF_INFO *sfinfo) Chris@40: { Chris@40: double seconds ; Chris@40: Chris@40: if (sfinfo->samplerate < 1) Chris@40: return NULL ; Chris@40: Chris@40: if (sfinfo->frames / sfinfo->samplerate > 0x7FFFFFFF) Chris@40: return "unknown" ; Chris@40: Chris@40: seconds = (1.0 * sfinfo->frames) / sfinfo->samplerate ; Chris@40: Chris@40: /* Accumulate the total of all known file durations */ Chris@40: total_seconds += seconds ; Chris@40: Chris@40: return format_duration_str (seconds) ; Chris@40: } /* generate_duration_str */ Chris@40: Chris@40: static void Chris@40: info_dump (const char *filename) Chris@40: { static char strbuffer [BUFFER_LEN] ; Chris@40: SNDFILE *file ; Chris@40: SF_INFO sfinfo ; Chris@40: double signal_max, decibels ; Chris@40: Chris@40: memset (&sfinfo, 0, sizeof (sfinfo)) ; Chris@40: Chris@40: if ((file = sf_open (filename, SFM_READ, &sfinfo)) == NULL) Chris@40: { printf ("Error : Not able to open input file %s.\n", filename) ; Chris@40: fflush (stdout) ; Chris@40: memset (data, 0, sizeof (data)) ; Chris@40: sf_command (file, SFC_GET_LOG_INFO, strbuffer, BUFFER_LEN) ; Chris@40: puts (strbuffer) ; Chris@40: puts (sf_strerror (NULL)) ; Chris@40: return ; Chris@40: } ; Chris@40: Chris@40: printf ("========================================\n") ; Chris@40: sf_command (file, SFC_GET_LOG_INFO, strbuffer, BUFFER_LEN) ; Chris@40: puts (strbuffer) ; Chris@40: printf ("----------------------------------------\n") ; Chris@40: Chris@40: printf ("Sample Rate : %d\n", sfinfo.samplerate) ; Chris@40: Chris@40: if (sfinfo.frames == SF_COUNT_MAX) Chris@40: printf ("Frames : unknown\n") ; Chris@40: else Chris@40: printf ("Frames : %" PRId64 "\n", sfinfo.frames) ; Chris@40: Chris@40: printf ("Channels : %d\n", sfinfo.channels) ; Chris@40: printf ("Format : 0x%08X\n", sfinfo.format) ; Chris@40: printf ("Sections : %d\n", sfinfo.sections) ; Chris@40: printf ("Seekable : %s\n", (sfinfo.seekable ? "TRUE" : "FALSE")) ; Chris@40: printf ("Duration : %s\n", generate_duration_str (&sfinfo)) ; Chris@40: Chris@40: if (sfinfo.frames < 100 * 1024 * 1024) Chris@40: { /* Do not use sf_signal_max because it doesn't work for non-seekable files . */ Chris@40: sf_command (file, SFC_CALC_SIGNAL_MAX, &signal_max, sizeof (signal_max)) ; Chris@40: decibels = calc_decibels (&sfinfo, signal_max) ; Chris@40: printf ("Signal Max : %g (%4.2f dB)\n", signal_max, decibels) ; Chris@40: } ; Chris@40: putchar ('\n') ; Chris@40: Chris@40: sf_close (file) ; Chris@40: Chris@40: } /* info_dump */ Chris@40: Chris@40: /*============================================================================== Chris@40: ** Dumping of SF_INSTRUMENT data. Chris@40: */ Chris@40: Chris@40: static const char * Chris@40: str_of_type (int mode) Chris@40: { switch (mode) Chris@40: { case SF_LOOP_NONE : return "none" ; Chris@40: case SF_LOOP_FORWARD : return "fwd " ; Chris@40: case SF_LOOP_BACKWARD : return "back" ; Chris@40: case SF_LOOP_ALTERNATING : return "alt " ; Chris@40: default : break ; Chris@40: } ; Chris@40: Chris@40: return "????" ; Chris@40: } /* str_of_mode */ Chris@40: Chris@40: static int Chris@40: instrument_dump (const char *filename) Chris@40: { SNDFILE *file ; Chris@40: SF_INFO sfinfo ; Chris@40: SF_INSTRUMENT inst ; Chris@40: int got_inst, k ; Chris@40: Chris@40: memset (&sfinfo, 0, sizeof (sfinfo)) ; Chris@40: Chris@40: if ((file = sf_open (filename, SFM_READ, &sfinfo)) == NULL) Chris@40: { printf ("Error : Not able to open input file %s.\n", filename) ; Chris@40: fflush (stdout) ; Chris@40: memset (data, 0, sizeof (data)) ; Chris@40: puts (sf_strerror (NULL)) ; Chris@40: return 1 ; Chris@40: } ; Chris@40: Chris@40: got_inst = sf_command (file, SFC_GET_INSTRUMENT, &inst, sizeof (inst)) ; Chris@40: sf_close (file) ; Chris@40: Chris@40: if (got_inst == SF_FALSE) Chris@40: { printf ("Error : File '%s' does not contain instrument data.\n\n", filename) ; Chris@40: return 1 ; Chris@40: } ; Chris@40: Chris@40: printf ("Instrument : %s\n\n", filename) ; Chris@40: printf (" Gain : %d\n", inst.gain) ; Chris@40: printf (" Base note : %d\n", inst.basenote) ; Chris@40: printf (" Velocity : %d - %d\n", (int) inst.velocity_lo, (int) inst.velocity_hi) ; Chris@40: printf (" Key : %d - %d\n", (int) inst.key_lo, (int) inst.key_hi) ; Chris@40: printf (" Loop points : %d\n", inst.loop_count) ; Chris@40: Chris@40: for (k = 0 ; k < inst.loop_count ; k++) Chris@40: printf (" %-2d Mode : %s Start : %6d End : %6d Count : %6d\n", k, str_of_type (inst.loops [k].mode), inst.loops [k].start, inst.loops [k].end, inst.loops [k].count) ; Chris@40: Chris@40: putchar ('\n') ; Chris@40: return 0 ; Chris@40: } /* instrument_dump */ Chris@40: Chris@40: static int Chris@40: broadcast_dump (const char *filename) Chris@40: { SNDFILE *file ; Chris@40: SF_INFO sfinfo ; Chris@40: SF_BROADCAST_INFO_2K bext ; Chris@40: double time_ref_sec ; Chris@40: int got_bext ; Chris@40: Chris@40: memset (&sfinfo, 0, sizeof (sfinfo)) ; Chris@40: Chris@40: if ((file = sf_open (filename, SFM_READ, &sfinfo)) == NULL) Chris@40: { printf ("Error : Not able to open input file %s.\n", filename) ; Chris@40: fflush (stdout) ; Chris@40: memset (data, 0, sizeof (data)) ; Chris@40: puts (sf_strerror (NULL)) ; Chris@40: return 1 ; Chris@40: } ; Chris@40: Chris@40: memset (&bext, 0, sizeof (SF_BROADCAST_INFO_2K)) ; Chris@40: Chris@40: got_bext = sf_command (file, SFC_GET_BROADCAST_INFO, &bext, sizeof (bext)) ; Chris@40: sf_close (file) ; Chris@40: Chris@40: if (got_bext == SF_FALSE) Chris@40: { printf ("Error : File '%s' does not contain broadcast information.\n\n", filename) ; Chris@40: return 1 ; Chris@40: } ; Chris@40: Chris@40: /* Chris@40: ** From : http://www.ebu.ch/en/technical/publications/userguides/bwf_user_guide.php Chris@40: ** Chris@40: ** Time Reference: Chris@40: ** This field is a count from midnight in samples to the first sample Chris@40: ** of the audio sequence. Chris@40: */ Chris@40: Chris@40: time_ref_sec = ((pow (2.0, 32) * bext.time_reference_high) + (1.0 * bext.time_reference_low)) / sfinfo.samplerate ; Chris@40: Chris@40: printf ("Description : %.*s\n", (int) sizeof (bext.description), bext.description) ; Chris@40: printf ("Originator : %.*s\n", (int) sizeof (bext.originator), bext.originator) ; Chris@40: printf ("Origination ref : %.*s\n", (int) sizeof (bext.originator_reference), bext.originator_reference) ; Chris@40: printf ("Origination date : %.*s\n", (int) sizeof (bext.origination_date), bext.origination_date) ; Chris@40: printf ("Origination time : %.*s\n", (int) sizeof (bext.origination_time), bext.origination_time) ; Chris@40: Chris@40: if (bext.time_reference_high == 0 && bext.time_reference_low == 0) Chris@40: printf ("Time ref : 0\n") ; Chris@40: else Chris@40: printf ("Time ref : 0x%x%08x (%.6f seconds)\n", bext.time_reference_high, bext.time_reference_low, time_ref_sec) ; Chris@40: Chris@40: printf ("BWF version : %d\n", bext.version) ; Chris@40: printf ("UMID : %.*s\n", (int) sizeof (bext.umid), bext.umid) ; Chris@40: printf ("Coding history : %.*s\n", bext.coding_history_size, bext.coding_history) ; Chris@40: Chris@40: return 0 ; Chris@40: } /* broadcast_dump */ Chris@40: Chris@40: static int Chris@40: chanmap_dump (const char *filename) Chris@40: { SNDFILE *file ; Chris@40: SF_INFO sfinfo ; Chris@40: int * channel_map ; Chris@40: int got_chanmap, k ; Chris@40: Chris@40: memset (&sfinfo, 0, sizeof (sfinfo)) ; Chris@40: Chris@40: if ((file = sf_open (filename, SFM_READ, &sfinfo)) == NULL) Chris@40: { printf ("Error : Not able to open input file %s.\n", filename) ; Chris@40: fflush (stdout) ; Chris@40: memset (data, 0, sizeof (data)) ; Chris@40: puts (sf_strerror (NULL)) ; Chris@40: return 1 ; Chris@40: } ; Chris@40: Chris@40: if ((channel_map = calloc (sfinfo.channels, sizeof (int))) == NULL) Chris@40: { printf ("Error : malloc failed.\n\n") ; Chris@40: return 1 ; Chris@40: } ; Chris@40: Chris@40: got_chanmap = sf_command (file, SFC_GET_CHANNEL_MAP_INFO, channel_map, sfinfo.channels * sizeof (int)) ; Chris@40: sf_close (file) ; Chris@40: Chris@40: if (got_chanmap == SF_FALSE) Chris@40: { printf ("Error : File '%s' does not contain channel map information.\n\n", filename) ; Chris@40: free (channel_map) ; Chris@40: return 1 ; Chris@40: } ; Chris@40: Chris@40: printf ("File : %s\n\n", filename) ; Chris@40: Chris@40: puts (" Chan Position") ; Chris@40: for (k = 0 ; k < sfinfo.channels ; k ++) Chris@40: { const char * name ; Chris@40: Chris@40: #define CASE_NAME(x) case x : name = #x ; break ; Chris@40: switch (channel_map [k]) Chris@40: { CASE_NAME (SF_CHANNEL_MAP_INVALID) ; Chris@40: CASE_NAME (SF_CHANNEL_MAP_MONO) ; Chris@40: CASE_NAME (SF_CHANNEL_MAP_LEFT) ; Chris@40: CASE_NAME (SF_CHANNEL_MAP_RIGHT) ; Chris@40: CASE_NAME (SF_CHANNEL_MAP_CENTER) ; Chris@40: CASE_NAME (SF_CHANNEL_MAP_FRONT_LEFT) ; Chris@40: CASE_NAME (SF_CHANNEL_MAP_FRONT_RIGHT) ; Chris@40: CASE_NAME (SF_CHANNEL_MAP_FRONT_CENTER) ; Chris@40: CASE_NAME (SF_CHANNEL_MAP_REAR_CENTER) ; Chris@40: CASE_NAME (SF_CHANNEL_MAP_REAR_LEFT) ; Chris@40: CASE_NAME (SF_CHANNEL_MAP_REAR_RIGHT) ; Chris@40: CASE_NAME (SF_CHANNEL_MAP_LFE) ; Chris@40: CASE_NAME (SF_CHANNEL_MAP_FRONT_LEFT_OF_CENTER) ; Chris@40: CASE_NAME (SF_CHANNEL_MAP_FRONT_RIGHT_OF_CENTER) ; Chris@40: CASE_NAME (SF_CHANNEL_MAP_SIDE_LEFT) ; Chris@40: CASE_NAME (SF_CHANNEL_MAP_SIDE_RIGHT) ; Chris@40: CASE_NAME (SF_CHANNEL_MAP_TOP_CENTER) ; Chris@40: CASE_NAME (SF_CHANNEL_MAP_TOP_FRONT_LEFT) ; Chris@40: CASE_NAME (SF_CHANNEL_MAP_TOP_FRONT_RIGHT) ; Chris@40: CASE_NAME (SF_CHANNEL_MAP_TOP_FRONT_CENTER) ; Chris@40: CASE_NAME (SF_CHANNEL_MAP_TOP_REAR_LEFT) ; Chris@40: CASE_NAME (SF_CHANNEL_MAP_TOP_REAR_RIGHT) ; Chris@40: CASE_NAME (SF_CHANNEL_MAP_TOP_REAR_CENTER) ; Chris@40: CASE_NAME (SF_CHANNEL_MAP_MAX) ; Chris@40: default : name = "default" ; Chris@40: break ; Chris@40: } ; Chris@40: Chris@40: printf (" %3d %s\n", k, name) ; Chris@40: } ; Chris@40: Chris@40: putchar ('\n') ; Chris@40: free (channel_map) ; Chris@40: Chris@40: return 0 ; Chris@40: } /* chanmap_dump */ Chris@40: Chris@40: static int Chris@40: cart_dump (const char *filename) Chris@40: { SNDFILE *file ; Chris@40: SF_INFO sfinfo ; Chris@40: SF_CART_INFO_VAR (1024) cart ; Chris@40: int got_cart, k ; Chris@40: Chris@40: memset (&sfinfo, 0, sizeof (sfinfo)) ; Chris@40: memset (&cart, 0, sizeof (cart)) ; Chris@40: Chris@40: if ((file = sf_open (filename, SFM_READ, &sfinfo)) == NULL) Chris@40: { printf ("Error : Not able to open input file %s.\n", filename) ; Chris@40: fflush (stdout) ; Chris@40: memset (data, 0, sizeof (data)) ; Chris@40: puts (sf_strerror (NULL)) ; Chris@40: return 1 ; Chris@40: } ; Chris@40: Chris@40: got_cart = sf_command (file, SFC_GET_CART_INFO, &cart, sizeof (cart)) ; Chris@40: sf_close (file) ; Chris@40: Chris@40: if (got_cart == SF_FALSE) Chris@40: { printf ("Error : File '%s' does not contain cart information.\n\n", filename) ; Chris@40: return 1 ; Chris@40: } ; Chris@40: Chris@40: printf ("Version : %.*s\n", (int) sizeof (cart.version), cart.version) ; Chris@40: printf ("Title : %.*s\n", (int) sizeof (cart.title), cart.title) ; Chris@40: printf ("Artist : %.*s\n", (int) sizeof (cart.artist), cart.artist) ; Chris@40: printf ("Cut id : %.*s\n", (int) sizeof (cart.cut_id), cart.cut_id) ; Chris@40: printf ("Category : %.*s\n", (int) sizeof (cart.category), cart.category) ; Chris@40: printf ("Classification : %.*s\n", (int) sizeof (cart.classification), cart.classification) ; Chris@40: printf ("Out cue : %.*s\n", (int) sizeof (cart.out_cue), cart.out_cue) ; Chris@40: printf ("Start date : %.*s\n", (int) sizeof (cart.start_date), cart.start_date) ; Chris@40: printf ("Start time : %.*s\n", (int) sizeof (cart.start_time), cart.start_time) ; Chris@40: printf ("End date : %.*s\n", (int) sizeof (cart.end_date), cart.end_date) ; Chris@40: printf ("End time : %.*s\n", (int) sizeof (cart.end_time), cart.end_time) ; Chris@40: printf ("App id : %.*s\n", (int) sizeof (cart.producer_app_id), cart.producer_app_id) ; Chris@40: printf ("App version : %.*s\n", (int) sizeof (cart.producer_app_version), cart.producer_app_version) ; Chris@40: printf ("User defined : %.*s\n", (int) sizeof (cart.user_def), cart.user_def) ; Chris@40: printf ("Level ref. : %d\n", cart.level_reference) ; Chris@40: printf ("Post timers :\n") ; Chris@40: Chris@40: for (k = 0 ; k < ARRAY_LEN (cart.post_timers) ; k++) Chris@40: if (cart.post_timers [k].usage [0]) Chris@40: printf (" %d %.*s %d\n", k, (int) sizeof (cart.post_timers [k].usage), cart.post_timers [k].usage, cart.post_timers [k].value) ; Chris@40: Chris@40: printf ("Reserved : %.*s\n", (int) sizeof (cart.reserved), cart.reserved) ; Chris@40: printf ("Url : %.*s\n", (int) sizeof (cart.url), cart.url) ; Chris@40: printf ("Tag text : %.*s\n", cart.tag_text_size, cart.tag_text) ; Chris@40: Chris@40: return 0 ; Chris@40: } /* cart_dump */ Chris@40: Chris@40: static void Chris@40: total_dump (void) Chris@40: { printf ("========================================\n") ; Chris@40: printf ("Total Duration : %s\n", format_duration_str (total_seconds)) ; Chris@40: } /* total_dump */