/* ** Copyright (C) 2006-2016 Erik de Castro Lopo ** Copyright (C) 2006 Paul Davis ** ** This program 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 program 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 program; if not, write to the Free Software ** Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA. */ #include "sfconfig.h" #include #include #include #include "common.h" static int gen_coding_history (char * added_history, int added_history_max, const SF_INFO * psfinfo) ; static inline size_t bc_min_size (const SF_BROADCAST_INFO* info) { if (info == NULL) return 0 ; return offsetof (SF_BROADCAST_INFO, coding_history) + info->coding_history_size ; } /* bc_min_size */ SF_BROADCAST_INFO_16K* broadcast_var_alloc (void) { return calloc (1, sizeof (SF_BROADCAST_INFO_16K)) ; } /* broadcast_var_alloc */ int broadcast_var_set (SF_PRIVATE *psf, const SF_BROADCAST_INFO * info, size_t datasize) { size_t len ; if (info == NULL) return SF_FALSE ; if (bc_min_size (info) > datasize) { psf->error = SFE_BAD_BROADCAST_INFO_SIZE ; return SF_FALSE ; } ; if (datasize >= sizeof (SF_BROADCAST_INFO_16K)) { psf->error = SFE_BAD_BROADCAST_INFO_TOO_BIG ; return SF_FALSE ; } ; if (psf->broadcast_16k == NULL) { if ((psf->broadcast_16k = broadcast_var_alloc ()) == NULL) { psf->error = SFE_MALLOC_FAILED ; return SF_FALSE ; } ; } ; /* Only copy the first part of the struct. */ memcpy (psf->broadcast_16k, info, offsetof (SF_BROADCAST_INFO, coding_history)) ; psf_strlcpy_crlf (psf->broadcast_16k->coding_history, info->coding_history, sizeof (psf->broadcast_16k->coding_history), datasize - offsetof (SF_BROADCAST_INFO, coding_history)) ; len = strlen (psf->broadcast_16k->coding_history) ; if (len > 0 && psf->broadcast_16k->coding_history [len - 1] != '\n') psf_strlcat (psf->broadcast_16k->coding_history, sizeof (psf->broadcast_16k->coding_history), "\r\n") ; if (psf->file.mode == SFM_WRITE) { char added_history [256] ; gen_coding_history (added_history, sizeof (added_history), &(psf->sf)) ; psf_strlcat (psf->broadcast_16k->coding_history, sizeof (psf->broadcast_16k->coding_history), added_history) ; } ; /* Force coding_history_size to be even. */ len = strlen (psf->broadcast_16k->coding_history) ; len += (len & 1) ? 1 : 0 ; psf->broadcast_16k->coding_history_size = (uint32_t) len ; /* Currently writing this version. */ psf->broadcast_16k->version = 2 ; return SF_TRUE ; } /* broadcast_var_set */ int broadcast_var_get (SF_PRIVATE *psf, SF_BROADCAST_INFO * data, size_t datasize) { size_t size ; if (psf->broadcast_16k == NULL) return SF_FALSE ; size = SF_MIN (datasize, bc_min_size ((const SF_BROADCAST_INFO *) psf->broadcast_16k)) ; memcpy (data, psf->broadcast_16k, size) ; return SF_TRUE ; } /* broadcast_var_get */ /*------------------------------------------------------------------------------ */ static int gen_coding_history (char * added_history, int added_history_max, const SF_INFO * psfinfo) { char chnstr [16] ; int count, width ; /* ** From : http://www.sr.se/utveckling/tu/bwf/docs/codhist2.htm ** ** Parameter Variable string Unit ** ========================================================================================== ** Coding Algorithm A= ** Sampling frequency F=<11000,22050,24000,32000,44100,48000> [Hz] ** Bit-rate B= ** Word Length W=<8, 12, 14, 16, 18, 20, 22, 24> [bits] ** Mode M= ** Text, free string T= */ switch (psfinfo->channels) { case 0 : return SF_FALSE ; case 1 : psf_strlcpy (chnstr, sizeof (chnstr), "mono") ; break ; case 2 : psf_strlcpy (chnstr, sizeof (chnstr), "stereo") ; break ; default : snprintf (chnstr, sizeof (chnstr), "%dchn", psfinfo->channels) ; break ; } ; switch (SF_CODEC (psfinfo->format)) { case SF_FORMAT_PCM_U8 : case SF_FORMAT_PCM_S8 : width = 8 ; break ; case SF_FORMAT_PCM_16 : width = 16 ; break ; case SF_FORMAT_PCM_24 : width = 24 ; break ; case SF_FORMAT_PCM_32 : width = 32 ; break ; case SF_FORMAT_FLOAT : width = 24 ; /* Bits in the mantissa + 1 */ break ; case SF_FORMAT_DOUBLE : width = 53 ; /* Bits in the mantissa + 1 */ break ; case SF_FORMAT_ULAW : case SF_FORMAT_ALAW : width = 12 ; break ; default : width = 42 ; break ; } ; count = snprintf (added_history, added_history_max, "A=PCM,F=%d,W=%d,M=%s,T=%s-%s\r\n", psfinfo->samplerate, width, chnstr, PACKAGE_NAME, PACKAGE_VERSION) ; if (count >= added_history_max) return 0 ; return count ; } /* gen_coding_history */