/[rdesktop]/sourceforge.net/trunk/rdesktop/rdpsnd_sgi.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_sgi.c

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

revision 746 by astrand, Mon Aug 9 13:03:32 2004 UTC revision 1346 by ossman_, Thu Dec 7 15:23:45 2006 UTC
# Line 1  Line 1 
1  /*  /* -*- c-basic-offset: 8 -*-
2     rdesktop: A Remote Desktop Protocol client.     rdesktop: A Remote Desktop Protocol client.
3     Sound Channel Process Functions - SGI/IRIX     Sound Channel Process Functions - SGI/IRIX
4     Copyright (C) Matthew Chapman 2003     Copyright (C) Matthew Chapman 2003
5     Copyright (C) GuoJunBo guojunbo@ict.ac.cn 2003     Copyright (C) GuoJunBo guojunbo@ict.ac.cn 2003
6     Copyright (C) Michael Gernoth mike@zerfleddert.de 2003     Copyright (C) Jeremy Meng void.foo@gmail.com 2004, 2005
7    
8     This program is free software; you can redistribute it and/or modify     This program is free software; you can redistribute it and/or modify
9     it under the terms of the GNU General Public License as published by     it under the terms of the GNU General Public License as published by
# Line 24  Line 24 
24  #include <errno.h>  #include <errno.h>
25  #include <dmedia/audio.h>  #include <dmedia/audio.h>
26    
27  #define IRIX_DEBUG 1  /* #define IRIX_DEBUG 1 */
28    
29  #define IRIX_MAX_VOL     65535  #define IRIX_MAX_VOL     65535
30    
 #define MAX_QUEUE       10  
   
 int g_dsp_fd;  
31  ALconfig audioconfig;  ALconfig audioconfig;
32  ALport output_port;  ALport output_port;
33    
 BOOL g_dsp_busy = False;  
 static BOOL g_swapaudio;  
34  static int g_snd_rate;  static int g_snd_rate;
 static BOOL g_swapaudio;  
 static short g_samplewidth;  
35  static int width = AL_SAMPLE_16;  static int width = AL_SAMPLE_16;
36    static char *sgi_output_device = NULL;
37    
38  double min_volume, max_volume, volume_range;  double min_volume, max_volume, volume_range;
39  int resource, maxFillable;  int resource, maxFillable;
40  int combinedFrameSize;  int combinedFrameSize;
41    
42  static struct audio_packet  void sgi_play(void);
43    
44    void
45    sgi_add_fds(int *n, fd_set * rfds, fd_set * wfds, struct timeval *tv)
46  {  {
47          struct stream s;  }
48          uint16 tick;  
49          uint8 index;  void
50  } packet_queue[MAX_QUEUE];  sgi_check_fds(fd_set * rfds, fd_set * wfds)
51  static unsigned int queue_hi, queue_lo;  {
52            if (output_port == (ALport) 0)
53                    return;
54    
55            if (!rdpsnd_queue_empty())
56                    sgi_play();
57    }
58    
59  BOOL  BOOL
60  wave_out_open(void)  sgi_open(void)
61  {  {
62          ALparamInfo pinfo;          ALparamInfo pinfo;
63            static int warned = 0;
64    
65  #if (defined(IRIX_DEBUG))  #if (defined(IRIX_DEBUG))
66          fprintf(stderr, "wave_out_open: begin\n");          fprintf(stderr, "sgi_open: begin\n");
67  #endif  #endif
68    
69            if (!warned && sgi_output_device)
70            {
71                    warning("device-options not supported for libao-driver\n");
72                    warned = 1;
73            }
74    
75          if (alGetParamInfo(AL_DEFAULT_OUTPUT, AL_GAIN, &pinfo) < 0)          if (alGetParamInfo(AL_DEFAULT_OUTPUT, AL_GAIN, &pinfo) < 0)
76          {          {
77                  fprintf(stderr, "wave_out_open: alGetParamInfo failed: %s\n",                  fprintf(stderr, "sgi_open: alGetParamInfo failed: %s\n",
78                          alGetErrorString(oserror()));                          alGetErrorString(oserror()));
79          }          }
80          min_volume = alFixedToDouble(pinfo.min.ll);          min_volume = alFixedToDouble(pinfo.min.ll);
81          max_volume = alFixedToDouble(pinfo.max.ll);          max_volume = alFixedToDouble(pinfo.max.ll);
82          volume_range = (max_volume - min_volume);          volume_range = (max_volume - min_volume);
83  #if (defined(IRIX_DEBUG))  #if (defined(IRIX_DEBUG))
84          fprintf(stderr, "wave_out_open: minvol = %lf, maxvol= %lf, range = %lf.\n",          fprintf(stderr, "sgi_open: minvol = %lf, maxvol= %lf, range = %lf.\n",
85                  min_volume, max_volume, volume_range);                  min_volume, max_volume, volume_range);
86  #endif  #endif
87    
         queue_lo = queue_hi = 0;  
   
88          audioconfig = alNewConfig();          audioconfig = alNewConfig();
89          if (audioconfig < 0)          if (audioconfig == (ALconfig) 0)
90          {          {
91                  fprintf(stderr, "wave_out_open: alNewConfig failed: %s\n",                  fprintf(stderr, "sgi_open: alNewConfig failed: %s\n", alGetErrorString(oserror()));
                         alGetErrorString(oserror()));  
92                  return False;                  return False;
93          }          }
94    
95          output_port = alOpenPort("rdpsnd", "w", 0);          output_port = alOpenPort("rdpsnd", "w", 0);
96          if (output_port == (ALport) 0)          if (output_port == (ALport) 0)
97          {          {
98                  fprintf(stderr, "wave_out_open: alOpenPort failed: %s\n",                  fprintf(stderr, "sgi_open: alOpenPort failed: %s\n", alGetErrorString(oserror()));
                         alGetErrorString(oserror()));  
99                  return False;                  return False;
100          }          }
101    
102  #if (defined(IRIX_DEBUG))  #if (defined(IRIX_DEBUG))
103          fprintf(stderr, "wave_out_open: returning\n");          fprintf(stderr, "sgi_open: returning\n");
104  #endif  #endif
105          return True;          return True;
106  }  }
107    
108  void  void
109  wave_out_close(void)  sgi_close(void)
110  {  {
111          /* Ack all remaining packets */          /* Ack all remaining packets */
112  #if (defined(IRIX_DEBUG))  #if (defined(IRIX_DEBUG))
113          fprintf(stderr, "wave_out_close: begin\n");          fprintf(stderr, "sgi_close: begin\n");
114  #endif  #endif
115    
116          while (queue_lo != queue_hi)          while (!rdpsnd_queue_empty())
117          {                  rdpsnd_queue_next(0);
                 rdpsnd_send_completion(packet_queue[queue_lo].tick, packet_queue[queue_lo].index);  
                 free(packet_queue[queue_lo].s.data);  
                 queue_lo = (queue_lo + 1) % MAX_QUEUE;  
         }  
118          alDiscardFrames(output_port, 0);          alDiscardFrames(output_port, 0);
119    
120          alClosePort(output_port);          alClosePort(output_port);
121            output_port = (ALport) 0;
122          alFreeConfig(audioconfig);          alFreeConfig(audioconfig);
123  #if (defined(IRIX_DEBUG))  #if (defined(IRIX_DEBUG))
124          fprintf(stderr, "wave_out_close: returning\n");          fprintf(stderr, "sgi_close: returning\n");
125  #endif  #endif
126  }  }
127    
128  BOOL  BOOL
129  wave_out_format_supported(WAVEFORMATEX * pwfx)  sgi_format_supported(WAVEFORMATEX * pwfx)
130  {  {
131          if (pwfx->wFormatTag != WAVE_FORMAT_PCM)          if (pwfx->wFormatTag != WAVE_FORMAT_PCM)
132                  return False;                  return False;
# Line 136  wave_out_format_supported(WAVEFORMATEX * Line 139  wave_out_format_supported(WAVEFORMATEX *
139  }  }
140    
141  BOOL  BOOL
142  wave_out_set_format(WAVEFORMATEX * pwfx)  sgi_set_format(WAVEFORMATEX * pwfx)
143  {  {
144          int channels;          int channels;
145          int frameSize, channelCount;          int frameSize, channelCount;
146          ALpv params;          ALpv params;
147    
148  #if (defined(IRIX_DEBUG))  #if (defined(IRIX_DEBUG))
149          fprintf(stderr, "wave_out_set_format: init...\n");          fprintf(stderr, "sgi_set_format: init...\n");
150  #endif  #endif
         /* limited support to configure an opened audio port in IRIX */  
         /* have to reopen the audio port, using same config */  
         alClosePort(output_port);  
   
         g_swapaudio = False;  
151    
152          if (pwfx->wBitsPerSample == 8)          if (pwfx->wBitsPerSample == 8)
153                  width = AL_SAMPLE_8;                  width = AL_SAMPLE_8;
154          else if (pwfx->wBitsPerSample == 16)          else if (pwfx->wBitsPerSample == 16)
         {  
155                  width = AL_SAMPLE_16;                  width = AL_SAMPLE_16;
                 /* Do we need to swap the 16bit values? (Are we BigEndian) */  
 #if (defined(IRIX_DEBUG))  
                 g_swapaudio = 1;  
 #else  
                 g_swapaudio = 0;  
 #endif  
         }  
156    
157          g_samplewidth = pwfx->wBitsPerSample / 8;          /* Limited support to configure an opened audio port in IRIX.  The
158               number of channels is a static setting and can not be changed after
159               a port is opened.  So if the number of channels remains the same, we
160               can configure other settings; otherwise we have to reopen the audio
161               port, using same config. */
162    
163          channels = pwfx->nChannels;          channels = pwfx->nChannels;
164          g_snd_rate = pwfx->nSamplesPerSec;          g_snd_rate = pwfx->nSamplesPerSec;
165    
166          alSetSampFmt(audioconfig, AL_SAMPFMT_TWOSCOMP);          alSetSampFmt(audioconfig, AL_SAMPFMT_TWOSCOMP);
167          alSetWidth(audioconfig, width);          alSetWidth(audioconfig, width);
168          alSetChannels(audioconfig, channels);          if (channels != alGetChannels(audioconfig))
169            {
170                    alClosePort(output_port);
171                    alSetChannels(audioconfig, channels);
172                    output_port = alOpenPort("rdpsnd", "w", audioconfig);
173    
174          output_port = alOpenPort("rdpsnd", "w", audioconfig);                  if (output_port == (ALport) 0)
175                    {
176                            fprintf(stderr, "sgi_set_format: alOpenPort failed: %s\n",
177                                    alGetErrorString(oserror()));
178                            return False;
179                    }
180    
         if (output_port == (ALport) 0)  
         {  
                 fprintf(stderr, "wave_out_set_format: alOpenPort failed: %s\n",  
                         alGetErrorString(oserror()));  
                 return False;  
181          }          }
182    
183          resource = alGetResource(output_port);          resource = alGetResource(output_port);
# Line 188  wave_out_set_format(WAVEFORMATEX * pwfx) Line 187  wave_out_set_format(WAVEFORMATEX * pwfx)
187    
188          if (frameSize == 0 || channelCount == 0)          if (frameSize == 0 || channelCount == 0)
189          {          {
190                  fprintf(stderr, "wave_out_set_format: bad frameSize or channelCount\n");                  fprintf(stderr, "sgi_set_format: bad frameSize or channelCount\n");
191                  return False;                  return False;
192          }          }
193          combinedFrameSize = frameSize * channelCount;          combinedFrameSize = frameSize * channelCount;
# Line 209  wave_out_set_format(WAVEFORMATEX * pwfx) Line 208  wave_out_set_format(WAVEFORMATEX * pwfx)
208          }          }
209    
210  #if (defined(IRIX_DEBUG))  #if (defined(IRIX_DEBUG))
211          fprintf(stderr, "wave_out_set_format: returning...\n");          fprintf(stderr, "sgi_set_format: returning...\n");
212  #endif  #endif
213          return True;          return True;
214  }  }
215    
216  void  void
217  wave_out_volume(uint16 left, uint16 right)  sgi_volume(uint16 left, uint16 right)
218  {  {
219          double gainleft, gainright;          double gainleft, gainright;
220          ALpv pv[1];          ALpv pv[1];
221          ALfixed gain[8];          ALfixed gain[8];
222    
223  #if (defined(IRIX_DEBUG))  #if (defined(IRIX_DEBUG))
224          fprintf(stderr, "wave_out_volume: begin\n");          fprintf(stderr, "sgi_volume: begin\n");
225          fprintf(stderr, "left='%d', right='%d'\n", left, right);          fprintf(stderr, "left='%d', right='%d'\n", left, right);
226  #endif  #endif
227    
# Line 237  wave_out_volume(uint16 left, uint16 righ Line 236  wave_out_volume(uint16 left, uint16 righ
236          pv[0].sizeIn = 8;          pv[0].sizeIn = 8;
237          if (alSetParams(AL_DEFAULT_OUTPUT, pv, 1) < 0)          if (alSetParams(AL_DEFAULT_OUTPUT, pv, 1) < 0)
238          {          {
239                  fprintf(stderr, "wave_out_volume: alSetParams failed: %s\n",                  fprintf(stderr, "sgi_volume: alSetParams failed: %s\n",
240                          alGetErrorString(oserror()));                          alGetErrorString(oserror()));
241                  return;                  return;
242          }          }
243    
244  #if (defined(IRIX_DEBUG))  #if (defined(IRIX_DEBUG))
245          fprintf(stderr, "wave_out_volume: returning\n");          fprintf(stderr, "sgi_volume: returning\n");
246  #endif  #endif
247  }  }
248    
249  void  void
250  wave_out_write(STREAM s, uint16 tick, uint8 index)  sgi_play(void)
 {  
         struct audio_packet *packet = &packet_queue[queue_hi];  
         unsigned int next_hi = (queue_hi + 1) % MAX_QUEUE;  
   
         if (next_hi == queue_lo)  
         {  
                 fprintf(stderr, "No space to queue audio packet\n");  
                 return;  
         }  
   
         queue_hi = next_hi;  
   
         packet->s = *s;  
         packet->tick = tick;  
         packet->index = index;  
         packet->s.p += 4;  
   
         /* we steal the data buffer from s, give it a new one */  
         s->data = malloc(s->size);  
   
         if (!g_dsp_busy)  
                 wave_out_play();  
 }  
   
 void  
 wave_out_play(void)  
251  {  {
252          struct audio_packet *packet;          struct audio_packet *packet;
253          ssize_t len;          ssize_t len;
254          unsigned int i;          unsigned int i;
         uint8 swap;  
255          STREAM out;          STREAM out;
         static long startedat_us;  
         static long startedat_s;  
         static BOOL started = False;  
         static BOOL swapped = False;  
         struct timeval tv;  
256          int gf;          int gf;
         static long long temp;  
257    
258          while (1)          while (1)
259          {          {
260                  if (queue_lo == queue_hi)                  if (rdpsnd_queue_empty())
                 {  
                         g_dsp_busy = False;  
261                          return;                          return;
                 }  
262    
263                  packet = &packet_queue[queue_lo];                  packet = rdpsnd_queue_current_packet();
264                  out = &packet->s;                  out = &packet->s;
265    
                 /* Swap the current packet, but only once */  
                 if (g_swapaudio && !swapped)  
                 {  
                         for (i = 0; i < out->end - out->p; i += 2)  
                         {  
                                 swap = *(out->p + i);  
                                 *(out->p + i) = *(out->p + i + 1);  
                                 *(out->p + i + 1) = swap;  
                         }  
                         swapped = True;  
                 }  
   
                 if (!started)  
                 {  
                         gettimeofday(&tv, NULL);  
                         startedat_us = tv.tv_usec;  
                         startedat_s = tv.tv_sec;  
                         started = True;  
                 }  
   
266                  len = out->end - out->p;                  len = out->end - out->p;
                 gf = alGetFillable(output_port);  
                 if (len > gf)  
                 {  
                         //len = gf * combinedFrameSize;  
 #if (defined(IRIX_DEBUG))  
                         //fprintf(stderr,"Fillable...\n");  
 #endif  
                 }  
267    
268                  alWriteFrames(output_port, out->p, len / combinedFrameSize);                  alWriteFrames(output_port, out->p, len / combinedFrameSize);
269    
270                  out->p += len;                  out->p += len;
271                  if (out->p == out->end)                  if (out->p == out->end)
272                  {                  {
273                          long long duration;                          gf = alGetFilled(output_port);
274                          long elapsed;                          if (gf < (4 * maxFillable / 10))
   
                         gettimeofday(&tv, NULL);  
                         duration = (out->size * (1000000 / (g_samplewidth * g_snd_rate)));  
                         elapsed = (tv.tv_sec - startedat_s) * 1000000 + (tv.tv_usec - startedat_us);  
                         /* 7/10 is not good for IRIX audio port, 4x/100 is suitable */  
                         if (elapsed >= (duration * 485) / 1000)  
275                          {                          {
276                                  rdpsnd_send_completion(packet->tick, packet->index);                                  rdpsnd_queue_next(0);
                                 free(out->data);  
                                 queue_lo = (queue_lo + 1) % MAX_QUEUE;  
                                 started = False;  
                                 swapped = False;  
277                          }                          }
278                          else                          else
279                          {                          {
280  #if (defined(IRIX_DEBUG))  #if (defined(IRIX_DEBUG))
281                                  //fprintf(stderr,"Busy playing...\n");  /*                              fprintf(stderr,"Busy playing...\n"); */
282  #endif  #endif
283                                  g_dsp_busy = True;                                  usleep(10);
284                                  return;                                  return;
285                          }                          }
286                  }                  }
287          }          }
288  }  }
289    
290    static struct audio_driver sgi_driver = {
291            .name = "sgi",
292            .description = "SGI output driver",
293    
294            .add_fds = sgi_add_fds,
295            .check_fds = sgi_check_fds,
296    
297            .wave_out_open = sgi_open,
298            .wave_out_close = sgi_close,
299            .wave_out_format_supported = sgi_format_supported,
300            .wave_out_set_format = sgi_set_format,
301            .wave_out_volume = sgi_volume,
302    
303            .need_byteswap_on_be = 1,
304            .need_resampling = 0,
305    };
306    
307    struct audio_driver *
308    sgi_register(char *options)
309    {
310            if (options)
311            {
312                    sgi_output_device = xstrdup(options);
313            }
314            return &sgi_driver;
315    }

Legend:
Removed from v.746  
changed lines
  Added in v.1346

  ViewVC Help
Powered by ViewVC 1.1.26