1 /*
2 ** Copyright (C) 2006-2016 Erik de Castro Lopo <erikd@mega-nerd.com>
3 ** Copyright (C) 2006 Paul Davis <paul@linuxaudiosystems.com>
4 **
5 ** This program is free software; you can redistribute it and/or modify
6 ** it under the terms of the GNU Lesser General Public License as published by
7 ** the Free Software Foundation; either version 2.1 of the License, or
8 ** (at your option) any later version.
9 **
10 ** This program is distributed in the hope that it will be useful,
11 ** but WITHOUT ANY WARRANTY; without even the implied warranty of
12 ** MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
13 ** GNU Lesser General Public License for more details.
14 **
15 ** You should have received a copy of the GNU Lesser General Public License
16 ** along with this program; if not, write to the Free Software
17 ** Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
18 */
19 
20 #include "sfconfig.h"
21 
22 #include <stdio.h>
23 #include <stddef.h>
24 #include <string.h>
25 
26 #include "common.h"
27 
28 
29 static int gen_coding_history (char * added_history, int added_history_max, const SF_INFO * psfinfo) ;
30 
31 static inline size_t
bc_min_size(const SF_BROADCAST_INFO * info)32 bc_min_size (const SF_BROADCAST_INFO* info)
33 {	if (info == NULL)
34 		return 0 ;
35 
36 	return offsetof (SF_BROADCAST_INFO, coding_history) + info->coding_history_size ;
37 } /* bc_min_size */
38 
39 SF_BROADCAST_INFO_16K*
broadcast_var_alloc(void)40 broadcast_var_alloc (void)
41 {	return calloc (1, sizeof (SF_BROADCAST_INFO_16K)) ;
42 } /* broadcast_var_alloc */
43 
44 int
broadcast_var_set(SF_PRIVATE * psf,const SF_BROADCAST_INFO * info,size_t datasize)45 broadcast_var_set (SF_PRIVATE *psf, const SF_BROADCAST_INFO * info, size_t datasize)
46 {	size_t len ;
47 
48 	if (info == NULL)
49 		return SF_FALSE ;
50 
51 	if (bc_min_size (info) > datasize)
52 	{	psf->error = SFE_BAD_BROADCAST_INFO_SIZE ;
53 		return SF_FALSE ;
54 		} ;
55 
56 	if (datasize >= sizeof (SF_BROADCAST_INFO_16K))
57 	{	psf->error = SFE_BAD_BROADCAST_INFO_TOO_BIG ;
58 		return SF_FALSE ;
59 		} ;
60 
61 	if (psf->broadcast_16k == NULL)
62 	{	if ((psf->broadcast_16k = broadcast_var_alloc ()) == NULL)
63 		{	psf->error = SFE_MALLOC_FAILED ;
64 			return SF_FALSE ;
65 			} ;
66 		} ;
67 
68 	/* Only copy the first part of the struct. */
69 	memcpy (psf->broadcast_16k, info, offsetof (SF_BROADCAST_INFO, coding_history)) ;
70 
71 	psf_strlcpy_crlf (psf->broadcast_16k->coding_history, info->coding_history, sizeof (psf->broadcast_16k->coding_history), datasize - offsetof (SF_BROADCAST_INFO, coding_history)) ;
72 	len = strlen (psf->broadcast_16k->coding_history) ;
73 
74 	if (len > 0 && psf->broadcast_16k->coding_history [len - 1] != '\n')
75 		psf_strlcat (psf->broadcast_16k->coding_history, sizeof (psf->broadcast_16k->coding_history), "\r\n") ;
76 
77 	if (psf->file.mode == SFM_WRITE)
78 	{	char added_history [256] ;
79 
80 		gen_coding_history (added_history, sizeof (added_history), &(psf->sf)) ;
81 		psf_strlcat (psf->broadcast_16k->coding_history, sizeof (psf->broadcast_16k->coding_history), added_history) ;
82 		} ;
83 
84 	/* Force coding_history_size to be even. */
85 	len = strlen (psf->broadcast_16k->coding_history) ;
86 	len += (len & 1) ? 1 : 0 ;
87 	psf->broadcast_16k->coding_history_size = len ;
88 
89 	/* Currently writing this version. */
90 	psf->broadcast_16k->version = 2 ;
91 
92 	return SF_TRUE ;
93 } /* broadcast_var_set */
94 
95 
96 int
broadcast_var_get(SF_PRIVATE * psf,SF_BROADCAST_INFO * data,size_t datasize)97 broadcast_var_get (SF_PRIVATE *psf, SF_BROADCAST_INFO * data, size_t datasize)
98 {	size_t size ;
99 
100 	if (psf->broadcast_16k == NULL)
101 		return SF_FALSE ;
102 
103 	size = SF_MIN (datasize, bc_min_size ((const SF_BROADCAST_INFO *) psf->broadcast_16k)) ;
104 
105 	memcpy (data, psf->broadcast_16k, size) ;
106 
107 	return SF_TRUE ;
108 } /* broadcast_var_get */
109 
110 /*------------------------------------------------------------------------------
111 */
112 
113 static int
gen_coding_history(char * added_history,int added_history_max,const SF_INFO * psfinfo)114 gen_coding_history (char * added_history, int added_history_max, const SF_INFO * psfinfo)
115 {	char chnstr [16] ;
116 	int count, width ;
117 
118 	/*
119 	**	From : http://www.sr.se/utveckling/tu/bwf/docs/codhist2.htm
120 	**
121 	**	Parameter            Variable string <allowed option>                 Unit
122 	**	==========================================================================================
123 	**	Coding Algorithm     A=<ANALOGUE, PCM, MPEG1L1, MPEG1L2, MPEG1L3,
124 	**	                    MPEG2L1, MPEG2L2, MPEG2L3>
125 	**	Sampling frequency   F=<11000,22050,24000,32000,44100,48000>          [Hz]
126 	**	Bit-rate             B=<any bit-rate allowed in MPEG 2 (ISO/IEC       [kbit/s per channel]
127 	**	                    13818-3)>
128 	**	Word Length          W=<8, 12, 14, 16, 18, 20, 22, 24>                [bits]
129 	**	Mode                 M=<mono, stereo, dual-mono, joint-stereo>
130 	**	Text, free string    T=<a free ASCII-text string for in house use.
131 	**	                    This string should contain no commas (ASCII
132 	**	                    2Chex). Examples of the contents: ID-No; codec
133 	**	                    type; A/D type>
134 	*/
135 
136 	switch (psfinfo->channels)
137 	{	case 0 :
138 			return SF_FALSE ;
139 
140 		case 1 :
141 			psf_strlcpy (chnstr, sizeof (chnstr), "mono") ;
142 			break ;
143 
144 		case 2 :
145 			psf_strlcpy (chnstr, sizeof (chnstr), "stereo") ;
146 			break ;
147 
148 		default :
149 			snprintf (chnstr, sizeof (chnstr), "%uchn", psfinfo->channels) ;
150 			break ;
151 		} ;
152 
153 	switch (SF_CODEC (psfinfo->format))
154 	{	case SF_FORMAT_PCM_U8 :
155 		case SF_FORMAT_PCM_S8 :
156 			width = 8 ;
157 			break ;
158 		case SF_FORMAT_PCM_16 :
159 			width = 16 ;
160 			break ;
161 		case SF_FORMAT_PCM_24 :
162 			width = 24 ;
163 			break ;
164 		case SF_FORMAT_PCM_32 :
165 			width = 32 ;
166 			break ;
167 		case SF_FORMAT_FLOAT :
168 			width = 24 ; /* Bits in the mantissa + 1 */
169 			break ;
170 		case SF_FORMAT_DOUBLE :
171 			width = 53 ; /* Bits in the mantissa + 1 */
172 			break ;
173 		case SF_FORMAT_ULAW :
174 		case SF_FORMAT_ALAW :
175 			width = 12 ;
176 			break ;
177 		default :
178 			width = 42 ;
179 			break ;
180 		} ;
181 
182 	count = snprintf (added_history, added_history_max,
183 							"A=PCM,F=%u,W=%d,M=%s,T=%s-%s\r\n",
184 							psfinfo->samplerate, width, chnstr, PACKAGE_NAME, PACKAGE_VERSION) ;
185 
186 	if (count >= added_history_max)
187 		return 0 ;
188 
189 	return count ;
190 } /* gen_coding_history */
191