1 // --------------------------------------------------------------------------
5 // Author: Guilhem Lavaux <lavaux@easynet.fr> (C) 1999, 2000
7 // --------------------------------------------------------------------------
9 #pragma implementation "sndcpcm.cpp"
12 #include "wx/wxprec.h"
24 #include "wx/mmedia/sndbase.h"
25 #include "wx/mmedia/sndpcm.h"
26 #include "wx/mmedia/sndcpcm.h"
28 wxSoundStreamPcm::wxSoundStreamPcm(wxSoundStream
& sndio
)
29 : wxSoundStreamCodec(sndio
)
32 m_function_out
= NULL
;
38 wxSoundStreamPcm::~wxSoundStreamPcm()
44 wxUint32
wxSoundStreamPcm::GetBestSize() const
49 // -----------------------------------------------------------------------
50 // "Converters" definitions/implementations
51 // -----------------------------------------------------------------------
53 #define DEFINE_CONV(name, input_type, output_type, convert) \
54 static void Convert_##name(const void *buf_in, void *buf_out, wxUint32 len) \
56 register input_type src; \
57 register const input_type *t_buf_in = (input_type *)buf_in; \
58 register output_type *t_buf_out = (output_type *)buf_out; \
62 *t_buf_out++ = convert; \
63 len -= sizeof(input_type); \
67 // TODO: define converters for all other formats (32, 24)
69 DEFINE_CONV(8_8_sign
, wxUint8
, wxUint8
, (src
^ 0x80))
71 DEFINE_CONV(8_16
, wxUint8
, wxUint16
, (((wxUint16
)src
) << 8))
72 DEFINE_CONV(8_16_swap
, wxUint8
, wxUint16
, (src
))
73 DEFINE_CONV(8_16_sign
, wxUint8
, wxUint16
, (((wxUint16
)(src
^ 0x80)) << 8))
74 DEFINE_CONV(8_16_sign_swap
, wxUint8
, wxUint16
, (src
^ 0x80))
76 DEFINE_CONV(16_8
, wxUint16
, wxUint8
, (wxUint8
)(src
>> 8))
77 DEFINE_CONV(16_8_sign
, wxUint16
, wxUint8
, (wxUint8
)((src
>> 8) ^ 0x80))
78 DEFINE_CONV(16_swap_8
, wxUint16
, wxUint8
, (wxUint8
)(src
& 0xff))
79 DEFINE_CONV(16_swap_8_sign
, wxUint16
, wxUint8
, (wxUint8
)((src
& 0xff) ^ 0x80))
81 //DEFINE_CONV(24_8, wxUint32, wxUint8, (wxUint8)(src >> 16))
82 //DEFINE_CONV(24_8_sig, wxUint32, wxUint8, (wxUint8)((src >> 16) ^ 0x80))
84 //DEFINE_CONV(32_8, wxUint32, wxUint8, (wxUint8)(src >> 24))
86 DEFINE_CONV(16_sign
, wxUint16
, wxUint16
, (src
^ 0x8000))
87 DEFINE_CONV(16_swap
, wxUint16
, wxUint16
, (((src
& 0xff) << 8) | ((src
>> 8) & 0xff)))
89 DEFINE_CONV(16_swap_16_sign
, wxUint16
, wxUint16
, ((((src
& 0xff) << 8) | ((src
>> 8) & 0xff)) ^ 0x80))
90 // DEFINE_CONV(16_sign_16_swap, wxUint16, wxUint16, ((((src & 0xff) << 8) | ((src >> 8) & 0xff)) ^ 0x8000))
91 DEFINE_CONV(16_swap_16_sign_swap
, wxUint16
, wxUint16
, (src
^ 0x80))
93 // -----------------------------------------------------------------------
94 // Main PCM stream converter table
95 // -----------------------------------------------------------------------
101 // XX swapped -> YY sign
103 // XX swapped -> YY swapped
104 // XX swapped -> YY swapped sign
106 // XX stereo -> YY mono
107 // XX stereo -> YY mono sign
109 // XX swapped stereo -> YY swapped mono
110 // XX swapped stereo -> YY swapped mono sign
112 // XX swapped stereo -> YY swapped mono
113 // XX swapped stereo -> YY swapped mono sign
115 static wxSoundStreamPcm::ConverterType s_converters
[4][3][2] = {
119 Convert_8_8_sign
/* 8 -> 8 sign */
132 Convert_8_16
, /* 8 -> 16 */
133 Convert_8_16_sign
/* 8 -> 16 sign */
136 Convert_8_16_swap
, /* 8 -> 16 swapped */
137 Convert_8_16_sign_swap
/* 8 -> 16 sign swapped */
146 Convert_16_8
, /* 16 -> 8 */
147 Convert_16_8_sign
/* 16 -> 8 sign */
150 Convert_16_swap_8
, /* 16 swapped -> 8 */
151 Convert_16_swap_8_sign
/* 16 swapped -> 8 sign */
162 Convert_16_sign
/* 16 -> 16 sign */
165 Convert_16_swap
, /* 16 swapped -> 16 */
166 Convert_16_swap_16_sign
/* 16 swapped -> 16 sign */
170 Convert_16_swap_16_sign_swap
/* 16 swapped -> 16 sign swapped */
175 // This is the buffer size multiplier. It gives the needed size of the output size.
176 static float s_converters_multip
[] = {1, 2, 0.5, 1};
179 // TODO: Read() and Write() aren't really safe. If you give it a buffer which
180 // is not aligned on 2, you may crash (See converter.def).
183 wxSoundStream
& wxSoundStreamPcm::Read(void *buffer
, wxUint32 len
)
187 // We must have a multiple of 2
190 if (!m_function_in
) {
191 m_sndio
->Read(buffer
, len
);
192 m_lastcount
= m_sndio
->GetLastAccess();
193 m_snderror
= m_sndio
->GetError();
197 in_bufsize
= GetReadSize(len
);
199 if (len
<= m_best_size
) {
200 m_sndio
->Read(m_prebuffer
, in_bufsize
);
201 m_snderror
= m_sndio
->GetError();
202 if (m_snderror
!= wxSOUND_NOERROR
) {
207 m_function_in(m_prebuffer
, buffer
, m_sndio
->GetLastAccess());
211 temp_buffer
= new char[in_bufsize
];
212 m_sndio
->Read(temp_buffer
, in_bufsize
);
214 m_snderror
= m_sndio
->GetError();
215 if (m_snderror
!= wxSOUND_NOERROR
) {
220 m_function_in(temp_buffer
, buffer
, m_sndio
->GetLastAccess());
222 delete[] temp_buffer
;
225 m_lastcount
= (wxUint32
)(m_sndio
->GetLastAccess() * m_multiplier_in
);
230 wxSoundStream
& wxSoundStreamPcm::Write(const void *buffer
, wxUint32 len
)
232 wxUint32 out_bufsize
;
234 if (!m_function_out
) {
235 m_sndio
->Write(buffer
, len
);
236 m_lastcount
= m_sndio
->GetLastAccess();
237 m_snderror
= m_sndio
->GetError();
241 out_bufsize
= GetWriteSize(len
);
243 if (len
<= m_best_size
) {
244 out_bufsize
= GetWriteSize(len
);
246 m_function_out(buffer
, m_prebuffer
, len
);
247 m_sndio
->Write(m_prebuffer
, out_bufsize
);
248 m_snderror
= m_sndio
->GetError();
249 if (m_snderror
!= wxSOUND_NOERROR
) {
256 temp_buffer
= new char[out_bufsize
];
257 m_function_out(buffer
, temp_buffer
, len
);
259 m_sndio
->Write(temp_buffer
, out_bufsize
);
260 m_snderror
= m_sndio
->GetError();
261 if (m_snderror
!= wxSOUND_NOERROR
) {
266 delete[] temp_buffer
;
269 m_lastcount
= (wxUint32
)(m_sndio
->GetLastAccess() / m_multiplier_out
);
274 bool wxSoundStreamPcm::SetSoundFormat(const wxSoundFormatBase
& format
)
276 wxSoundFormatBase
*new_format
;
277 wxSoundFormatPcm
*pcm_format
, *pcm_format2
;
279 if (m_sndio
->SetSoundFormat(format
)) {
280 m_function_out
= NULL
;
281 m_function_in
= NULL
;
284 if (format
.GetType() != wxSOUND_PCM
) {
285 m_snderror
= wxSOUND_INVFRMT
;
291 new_format
= m_sndio
->GetSoundFormat().Clone();
292 pcm_format
= (wxSoundFormatPcm
*)&format
;
293 pcm_format2
= (wxSoundFormatPcm
*)new_format
;
296 // ----------------------------------------------------
297 // Test whether we need to resample
298 if (pcm_format
->GetSampleRate() != pcm_format2
->GetSampleRate()) {
299 wxUint32 src_rate
, dst_rate
;
301 src_rate
= pcm_format
->GetSampleRate();
302 dst_rate
= pcm_format2
->GetSampleRate();
303 m_needResampling
= true;
304 if (src_rate
< dst_rate
)
305 m_expandSamples
= true;
307 m_expandSamples
= false;
308 m_pitch
= (src_rate
<< FLOATBITS
) / dst_rate
;
311 // ----------------------------------------------------
312 // Select table to use:
313 // * 8 bits -> 8 bits
314 // * 16 bits -> 8 bits
315 // * 8 bits -> 16 bits
316 // * 16 bits -> 16 bits
318 int table_no
, table_no2
;
321 switch (pcm_format
->GetBPS()) {
329 // TODO: Add something here: error, log, ...
332 switch (pcm_format2
->GetBPS()) {
340 // TODO: Add something here: error, log, ...
344 if (pcm_format2
->Signed() != pcm_format
->Signed())
349 #define MY_ORDER wxBYTE_ORDER
350 #if wxBYTE_ORDER == wxLITTLE_ENDIAN
351 #define OTHER_ORDER wxBIG_ENDIAN
353 #define OTHER_ORDER wxLITTLE_ENDIAN
356 // --------------------------------------------------------
357 // Find the good converter !
359 if (pcm_format
->GetOrder() == OTHER_ORDER
) {
360 if (pcm_format
->GetOrder() == pcm_format2
->GetOrder())
365 if (pcm_format
->GetOrder() == pcm_format2
->GetOrder())
371 m_function_out
= s_converters
[table_no
*2+table_no2
][i_swap
][i_sign
];
372 m_function_in
= s_converters
[table_no2
*2+table_no
][i_swap
][i_sign
];
373 m_multiplier_out
= s_converters_multip
[table_no
*2+table_no2
];
374 m_multiplier_in
= s_converters_multip
[table_no2
*2+table_no2
];
377 delete[] m_prebuffer
;
379 // We try to minimize the need of dynamic memory allocation by preallocating a buffer. But
380 // to be sure it will be efficient we minimize the best size.
381 if (m_multiplier_in
< m_multiplier_out
) {
382 m_prebuffer_size
= (wxUint32
)(m_sndio
->GetBestSize() *
384 m_best_size
= (wxUint32
)(m_sndio
->GetBestSize() *
387 m_prebuffer_size
= (wxUint32
)(m_sndio
->GetBestSize() *
389 m_best_size
= (wxUint32
)(m_sndio
->GetBestSize() *
393 m_prebuffer
= new char[m_prebuffer_size
];
395 bool SetSoundFormatReturn
;
397 SetSoundFormatReturn
= m_sndio
->SetSoundFormat(*new_format
);
398 wxASSERT( SetSoundFormatReturn
);
399 wxUnusedVar( SetSoundFormatReturn
);
401 m_sndformat
= new_format
;
405 wxUint32
wxSoundStreamPcm::GetWriteSize(wxUint32 len
) const
407 // For the moment, it is simple but next time it will become more complicated
409 return (wxUint32
)(len
* m_multiplier_out
);
412 wxUint32
wxSoundStreamPcm::GetReadSize(wxUint32 len
) const
414 return (wxUint32
)(len
/ m_multiplier_in
);
417 // Resampling engine. NOT FINISHED and NOT INCLUDED but this is a first DRAFT.
423 #define FLOATMASK 0xffff
424 #define INTMASK 0xffff0000
426 void ResamplingShrink_
##DEPTH##(const void *source, void *destination, wxUint32 len)
428 wxUint
##DEPTH## *source_data, *dest_data;
431 source_data
= (wxUint
##DEPTH## *)source;
432 dest_data
= (wxUint
##DEPTH## *)destination;
436 // Increment the position in the input buffer
441 *dest_data
++ = *source_data
;