/[rdesktop]/sourceforge.net/trunk/rdesktop/rdpsnd_sun.c
This is repository of my old source code which isn't updated any more. Go to git.rot13.org for current projects!
ViewVC logotype

Diff of /sourceforge.net/trunk/rdesktop/rdpsnd_sun.c

Parent Directory Parent Directory | Revision Log Revision Log | View Patch Patch

revision 476 by matthewc, Sat Oct 4 00:03:24 2003 UTC revision 538 by stargo, Fri Oct 31 09:38:25 2003 UTC
# Line 25  Line 25 
25  #include <fcntl.h>  #include <fcntl.h>
26  #include <errno.h>  #include <errno.h>
27  #include <sys/ioctl.h>  #include <sys/ioctl.h>
28  #include <sys/audio.h>  #include <sys/audioio.h>
29  #include <stropts.h>  #include <stropts.h>
30    
31  #define MAX_QUEUE       10  #define MAX_QUEUE       10
32    
33  int g_dsp_fd;  int g_dsp_fd;
34  BOOL g_dsp_busy;  BOOL g_dsp_busy = False;
35  static BOOL swapaudio;  static BOOL g_reopened;
36  static short samplewidth;  static BOOL g_swapaudio;
37    static short g_samplewidth;
38    
39  static struct audio_packet {  static struct audio_packet
40    {
41          struct stream s;          struct stream s;
42          uint16 tick;          uint16 tick;
43          uint8 index;          uint8 index;
# Line 45  static unsigned int queue_hi, queue_lo; Line 47  static unsigned int queue_hi, queue_lo;
47  BOOL  BOOL
48  wave_out_open(void)  wave_out_open(void)
49  {  {
50          char *dsp_dev = "/dev/audio";          char *dsp_dev = getenv("AUDIODEV");
51    
52            if (dsp_dev == NULL)
53            {
54                    dsp_dev = "/dev/audio";
55            }
56    
57          if ((g_dsp_fd = open(dsp_dev, O_WRONLY|O_NONBLOCK)) == -1)          if ((g_dsp_fd = open(dsp_dev, O_WRONLY | O_NONBLOCK)) == -1)
58          {          {
59                  perror(dsp_dev);                  perror(dsp_dev);
60                  return False;                  return False;
61          }          }
62    
   
63          /* Non-blocking so that user interface is responsive */          /* Non-blocking so that user interface is responsive */
64          fcntl(g_dsp_fd, F_SETFL, fcntl(g_dsp_fd, F_GETFL)|O_NONBLOCK);          fcntl(g_dsp_fd, F_SETFL, fcntl(g_dsp_fd, F_GETFL) | O_NONBLOCK);
65    
66          queue_lo = queue_hi = 0;          queue_lo = queue_hi = 0;
67            g_reopened = True;
68    
69          return True;          return True;
70  }  }
71    
72  void  void
73  wave_out_close(void)  wave_out_close(void)
74  {  {
75          ioctl(g_dsp_fd,I_FLUSH,FLUSHW);          /* Ack all remaining packets */
76            while (queue_lo != queue_hi)
77            {
78                    rdpsnd_send_completion(packet_queue[queue_lo].tick, packet_queue[queue_lo].index);
79                    free(packet_queue[queue_lo].s.data);
80                    queue_lo = (queue_lo + 1) % MAX_QUEUE;
81            }
82    
83    #if defined I_FLUSH && defined FLUSHW
84            /* Flush the audiobuffer */
85            ioctl(g_dsp_fd, I_FLUSH, FLUSHW);
86    #endif
87          close(g_dsp_fd);          close(g_dsp_fd);
88  }  }
89    
90  BOOL  BOOL
91  wave_out_format_supported(WAVEFORMATEX *pwfx)  wave_out_format_supported(WAVEFORMATEX * pwfx)
92  {  {
93          if (pwfx->wFormatTag != WAVE_FORMAT_PCM)          if (pwfx->wFormatTag != WAVE_FORMAT_PCM)
94                  return False;                  return False;
# Line 81  wave_out_format_supported(WAVEFORMATEX * Line 101  wave_out_format_supported(WAVEFORMATEX *
101  }  }
102    
103  BOOL  BOOL
104  wave_out_set_format(WAVEFORMATEX *pwfx)  wave_out_set_format(WAVEFORMATEX * pwfx)
105  {  {
106          audio_info_t info;          audio_info_t info;
107          int test = 1;          int test = 1;
108    
109          ioctl(g_dsp_fd, AUDIO_DRAIN, 0);          ioctl(g_dsp_fd, AUDIO_DRAIN, 0);
110          swapaudio = False;          g_swapaudio = False;
111          AUDIO_INITINFO(&info);          AUDIO_INITINFO(&info);
112    
113    
114          if (pwfx->wBitsPerSample == 8)          if (pwfx->wBitsPerSample == 8)
115          {          {
116                  info.play.encoding = AUDIO_ENCODING_LINEAR8;                  info.play.encoding = AUDIO_ENCODING_LINEAR8;
                 samplewidth=1;  
117          }          }
118          else if (pwfx->wBitsPerSample == 16)          else if (pwfx->wBitsPerSample == 16)
119          {          {
120                  info.play.encoding = AUDIO_ENCODING_LINEAR;                  info.play.encoding = AUDIO_ENCODING_LINEAR;
121                  swapaudio = !(*(uint8 *) (&test));                  /* Do we need to swap the 16bit values? (Are we BigEndian) */
122                  samplewidth=2;                  g_swapaudio = !(*(uint8 *) (&test));
123          }          }
124    
125          if (pwfx->nChannels == 1 )          g_samplewidth = pwfx->wBitsPerSample / 8;
126          {        
127                  info.play.channels = AUDIO_CHANNELS_MONO;          if (pwfx->nChannels == 1)
128            {
129                    info.play.channels = 1;
130          }          }
131          else if (pwfx->nChannels == 2 )          else if (pwfx->nChannels == 2)
132          {          {
133                  info.play.channels = AUDIO_CHANNELS_STEREO;                  info.play.channels = 2;
134                  samplewidth*=2;                  g_samplewidth *= 2;
135          }          }
136    
137          info.play.sample_rate = pwfx->nSamplesPerSec;          info.play.sample_rate = pwfx->nSamplesPerSec;
# Line 118  wave_out_set_format(WAVEFORMATEX *pwfx) Line 139  wave_out_set_format(WAVEFORMATEX *pwfx)
139          info.play.samples = 0;          info.play.samples = 0;
140          info.play.eof = 0;          info.play.eof = 0;
141          info.play.error = 0;          info.play.error = 0;
142            g_reopened = True;
143    
144          if (ioctl(g_dsp_fd, AUDIO_SETINFO, &info) == -1)          if (ioctl(g_dsp_fd, AUDIO_SETINFO, &info) == -1)
145          {          {
# Line 130  wave_out_set_format(WAVEFORMATEX *pwfx) Line 152  wave_out_set_format(WAVEFORMATEX *pwfx)
152  }  }
153    
154  void  void
155    wave_out_volume(uint16 left, uint16 right)
156    {
157            audio_info_t info;
158            uint balance;
159            uint volume;
160    
161            if (ioctl(g_dsp_fd, AUDIO_GETINFO, &info) == -1)
162            {
163                    perror("AUDIO_GETINFO");
164                    return;
165            }
166    
167            volume = (left > right) ? left : right;
168    
169            if (volume / AUDIO_MID_BALANCE != 0)
170            {
171                    balance =
172                            AUDIO_MID_BALANCE - (left / (volume / AUDIO_MID_BALANCE)) +
173                            (right / (volume / AUDIO_MID_BALANCE));
174            }
175            else
176            {
177                    balance = AUDIO_MID_BALANCE;
178            }
179    
180            info.play.gain = volume / (65536 / AUDIO_MAX_GAIN);
181            info.play.balance = balance;
182    
183            if (ioctl(g_dsp_fd, AUDIO_SETINFO, &info) == -1)
184            {
185                    perror("AUDIO_SETINFO");
186                    return;
187            }
188    }
189    
190    void
191  wave_out_write(STREAM s, uint16 tick, uint8 index)  wave_out_write(STREAM s, uint16 tick, uint8 index)
192  {  {
193          struct audio_packet *packet = &packet_queue[queue_hi];          struct audio_packet *packet = &packet_queue[queue_hi];
# Line 140  wave_out_write(STREAM s, uint16 tick, ui Line 198  wave_out_write(STREAM s, uint16 tick, ui
198                  error("No space to queue audio packet\n");                  error("No space to queue audio packet\n");
199                  return;                  return;
200          }          }
201            
202          queue_hi = next_hi;          queue_hi = next_hi;
203    
204          packet->s = *s;          packet->s = *s;
205          packet->tick = tick;          packet->tick = tick;
206          packet->index = index;          packet->index = index;
207            packet->s.p += 4;
208    
209          /* we steal the data buffer from s, give it a new one */          /* we steal the data buffer from s, give it a new one */
210          s->data = malloc(s->size);          s->data = malloc(s->size);
# Line 165  wave_out_play(void) Line 224  wave_out_play(void)
224          STREAM out;          STREAM out;
225          static BOOL swapped = False;          static BOOL swapped = False;
226          static BOOL sentcompletion = True;          static BOOL sentcompletion = True;
227          static int samplecnt;          static uint32 samplecnt = 0;
228          static int numsamples;          static uint32 numsamples;
229    
230          while (1)          while (1)
231          {          {
232                    if (g_reopened)
233                    {
234                            /* Device was just (re)openend */
235                            samplecnt = 0;
236                            swapped = False;
237                            sentcompletion = True;
238                            g_reopened = False;
239                    }
240    
241                  if (queue_lo == queue_hi)                  if (queue_lo == queue_hi)
242                  {                  {
243                          g_dsp_busy = 0;                          g_dsp_busy = 0;
# Line 179  wave_out_play(void) Line 247  wave_out_play(void)
247                  packet = &packet_queue[queue_lo];                  packet = &packet_queue[queue_lo];
248                  out = &packet->s;                  out = &packet->s;
249    
250                  if ( swapaudio && ! swapped )                  /* Swap the current packet, but only once */
251                    if (g_swapaudio && !swapped)
252                  {                  {
253                          for ( i = 0; i < out->end - out->p; i+=2 )                          for (i = 0; i < out->end - out->p; i += 2)
254                          {                          {
255                                  swap = *(out->p + i);                                  swap = *(out->p + i);
256                                  *(out->p + i ) = *(out->p + i + 1);                                  *(out->p + i) = *(out->p + i + 1);
257                                  *(out->p + i + 1) = swap;                                  *(out->p + i + 1) = swap;
                                 swapped = True;  
258                          }                          }
259                            swapped = True;
260                  }                  }
261    
262                  if ( sentcompletion )                  if (sentcompletion)
263                  {                  {
                         if (ioctl(g_dsp_fd, AUDIO_GETINFO, &info) == -1)  
                         {  
                                 perror("AUDIO_GETINFO");  
                                 return;  
                         }  
                         samplecnt=info.play.samples;  
264                          sentcompletion = False;                          sentcompletion = False;
265                          numsamples = (out->end-out->p)/samplewidth;                          numsamples = (out->end - out->p) / g_samplewidth;
266                  }                  }
267    
268                  len=0;                  len = 0;
269    
270                  if ( out->end - out->p != 0 )                  if (out->end != out->p)
271                  {                  {
272                          len = write(g_dsp_fd, out->p, out->end - out->p);                          len = write(g_dsp_fd, out->p, out->end - out->p);
273                          if (len == -1)                          if (len == -1)
# Line 224  wave_out_play(void) Line 287  wave_out_play(void)
287                                  perror("AUDIO_GETINFO");                                  perror("AUDIO_GETINFO");
288                                  return;                                  return;
289                          }                          }
290                          if ( info.play.samples >= samplecnt+(numsamples)*0.9 )  
291                            /* Ack the packet, if we have played at least 70% */
292                            if (info.play.samples >= samplecnt + ((numsamples * 7) / 10))
293                          {                          {
294                                    samplecnt += numsamples;
295                                  rdpsnd_send_completion(packet->tick, packet->index);                                  rdpsnd_send_completion(packet->tick, packet->index);
296                                  free(out->data);                                  free(out->data);
297                                  queue_lo = (queue_lo + 1) % MAX_QUEUE;                                  queue_lo = (queue_lo + 1) % MAX_QUEUE;

Legend:
Removed from v.476  
changed lines
  Added in v.538

  ViewVC Help
Powered by ViewVC 1.1.26