]>
Commit | Line | Data |
---|---|---|
85571bc7 FB |
1 | /* |
2 | * QEMU Mixing engine | |
1d14ffa9 FB |
3 | * |
4 | * Copyright (c) 2004-2005 Vassili Karpov (malc) | |
5 | * | |
85571bc7 FB |
6 | * Permission is hereby granted, free of charge, to any person obtaining a copy |
7 | * of this software and associated documentation files (the "Software"), to deal | |
8 | * in the Software without restriction, including without limitation the rights | |
9 | * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell | |
10 | * copies of the Software, and to permit persons to whom the Software is | |
11 | * furnished to do so, subject to the following conditions: | |
12 | * | |
13 | * The above copyright notice and this permission notice shall be included in | |
14 | * all copies or substantial portions of the Software. | |
15 | * | |
16 | * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR | |
17 | * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, | |
18 | * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL | |
19 | * THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER | |
20 | * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, | |
21 | * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN | |
22 | * THE SOFTWARE. | |
23 | */ | |
24 | ||
25 | /* | |
26 | * Tusen tack till Mike Nordell | |
27 | * dec++'ified by Dscho | |
28 | */ | |
29 | ||
1d14ffa9 FB |
30 | #ifndef SIGNED |
31 | #define HALF (IN_MAX >> 1) | |
32 | #endif | |
33 | ||
a2885387 RPM |
34 | #define ET glue (ENDIAN_CONVERSION, glue (glue (glue (_, ITYPE), BSIZE), _t)) |
35 | #define IN_T glue (glue (ITYPE, BSIZE), _t) | |
1d14ffa9 FB |
36 | |
37 | #ifdef FLOAT_MIXENG | |
203cea22 | 38 | static inline mixeng_real glue (conv_, ET) (IN_T v) |
1d14ffa9 FB |
39 | { |
40 | IN_T nv = ENDIAN_CONVERT (v); | |
41 | ||
42 | #ifdef RECIPROCAL | |
43 | #ifdef SIGNED | |
1ea879e5 | 44 | return nv * (1.f / (mixeng_real) (IN_MAX - IN_MIN)); |
1d14ffa9 | 45 | #else |
1ea879e5 | 46 | return (nv - HALF) * (1.f / (mixeng_real) IN_MAX); |
1d14ffa9 FB |
47 | #endif |
48 | #else /* !RECIPROCAL */ | |
85571bc7 | 49 | #ifdef SIGNED |
578c7b2c | 50 | return nv / (mixeng_real) ((mixeng_real) IN_MAX - IN_MIN); |
85571bc7 | 51 | #else |
1ea879e5 | 52 | return (nv - HALF) / (mixeng_real) IN_MAX; |
85571bc7 | 53 | #endif |
1d14ffa9 FB |
54 | #endif |
55 | } | |
85571bc7 | 56 | |
203cea22 | 57 | static inline IN_T glue (clip_, ET) (mixeng_real v) |
85571bc7 | 58 | { |
1d14ffa9 FB |
59 | if (v >= 0.5) { |
60 | return IN_MAX; | |
61 | } | |
62 | else if (v < -0.5) { | |
63 | return IN_MIN; | |
64 | } | |
65 | ||
66 | #ifdef SIGNED | |
578c7b2c | 67 | return ENDIAN_CONVERT ((IN_T) (v * ((mixeng_real) IN_MAX - IN_MIN))); |
1d14ffa9 FB |
68 | #else |
69 | return ENDIAN_CONVERT ((IN_T) ((v * IN_MAX) + HALF)); | |
70 | #endif | |
71 | } | |
72 | ||
73 | #else /* !FLOAT_MIXENG */ | |
74 | ||
75 | static inline int64_t glue (conv_, ET) (IN_T v) | |
76 | { | |
77 | IN_T nv = ENDIAN_CONVERT (v); | |
85571bc7 | 78 | #ifdef SIGNED |
1d14ffa9 | 79 | return ((int64_t) nv) << (32 - SHIFT); |
85571bc7 | 80 | #else |
1d14ffa9 | 81 | return ((int64_t) nv - HALF) << (32 - SHIFT); |
85571bc7 FB |
82 | #endif |
83 | } | |
84 | ||
1d14ffa9 | 85 | static inline IN_T glue (clip_, ET) (int64_t v) |
85571bc7 | 86 | { |
1d14ffa9 | 87 | if (v >= 0x7f000000) { |
85571bc7 | 88 | return IN_MAX; |
1d14ffa9 FB |
89 | } |
90 | else if (v < -2147483648LL) { | |
85571bc7 | 91 | return IN_MIN; |
1d14ffa9 | 92 | } |
85571bc7 FB |
93 | |
94 | #ifdef SIGNED | |
1d14ffa9 | 95 | return ENDIAN_CONVERT ((IN_T) (v >> (32 - SHIFT))); |
85571bc7 | 96 | #else |
1d14ffa9 | 97 | return ENDIAN_CONVERT ((IN_T) ((v >> (32 - SHIFT)) + HALF)); |
85571bc7 FB |
98 | #endif |
99 | } | |
1d14ffa9 | 100 | #endif |
85571bc7 | 101 | |
1d14ffa9 | 102 | static void glue (glue (conv_, ET), _to_stereo) |
00e07679 | 103 | (struct st_sample *dst, const void *src, int samples) |
85571bc7 | 104 | { |
1ea879e5 | 105 | struct st_sample *out = dst; |
85571bc7 | 106 | IN_T *in = (IN_T *) src; |
00e07679 | 107 | |
85571bc7 | 108 | while (samples--) { |
00e07679 MW |
109 | out->l = glue (conv_, ET) (*in++); |
110 | out->r = glue (conv_, ET) (*in++); | |
85571bc7 FB |
111 | out += 1; |
112 | } | |
113 | } | |
114 | ||
1d14ffa9 | 115 | static void glue (glue (conv_, ET), _to_mono) |
00e07679 | 116 | (struct st_sample *dst, const void *src, int samples) |
85571bc7 | 117 | { |
1ea879e5 | 118 | struct st_sample *out = dst; |
85571bc7 | 119 | IN_T *in = (IN_T *) src; |
00e07679 | 120 | |
85571bc7 | 121 | while (samples--) { |
00e07679 | 122 | out->l = glue (conv_, ET) (in[0]); |
85571bc7 FB |
123 | out->r = out->l; |
124 | out += 1; | |
125 | in += 1; | |
126 | } | |
127 | } | |
128 | ||
1d14ffa9 | 129 | static void glue (glue (clip_, ET), _from_stereo) |
1ea879e5 | 130 | (void *dst, const struct st_sample *src, int samples) |
85571bc7 | 131 | { |
1ea879e5 | 132 | const struct st_sample *in = src; |
85571bc7 FB |
133 | IN_T *out = (IN_T *) dst; |
134 | while (samples--) { | |
1d14ffa9 FB |
135 | *out++ = glue (clip_, ET) (in->l); |
136 | *out++ = glue (clip_, ET) (in->r); | |
85571bc7 FB |
137 | in += 1; |
138 | } | |
139 | } | |
140 | ||
1d14ffa9 | 141 | static void glue (glue (clip_, ET), _from_mono) |
1ea879e5 | 142 | (void *dst, const struct st_sample *src, int samples) |
85571bc7 | 143 | { |
1ea879e5 | 144 | const struct st_sample *in = src; |
85571bc7 FB |
145 | IN_T *out = (IN_T *) dst; |
146 | while (samples--) { | |
1d14ffa9 | 147 | *out++ = glue (clip_, ET) (in->l + in->r); |
85571bc7 FB |
148 | in += 1; |
149 | } | |
150 | } | |
151 | ||
1d14ffa9 | 152 | #undef ET |
85571bc7 | 153 | #undef HALF |
a2885387 | 154 | #undef IN_T |