/[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 1345 by ossman_, Thu Dec 7 11:54:29 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    
 static struct audio_packet  
 {  
         struct stream s;  
         uint16 tick;  
         uint8 index;  
 } packet_queue[MAX_QUEUE];  
 static unsigned int queue_hi, queue_lo;  
   
42  BOOL  BOOL
43  wave_out_open(void)  sgi_open(void)
44  {  {
45          ALparamInfo pinfo;          ALparamInfo pinfo;
46            static int warned = 0;
47    
48  #if (defined(IRIX_DEBUG))  #if (defined(IRIX_DEBUG))
49          fprintf(stderr, "wave_out_open: begin\n");          fprintf(stderr, "sgi_open: begin\n");
50  #endif  #endif
51    
52            if (!warned && sgi_output_device)
53            {
54                    warning("device-options not supported for libao-driver\n");
55                    warned = 1;
56            }
57    
58          if (alGetParamInfo(AL_DEFAULT_OUTPUT, AL_GAIN, &pinfo) < 0)          if (alGetParamInfo(AL_DEFAULT_OUTPUT, AL_GAIN, &pinfo) < 0)
59          {          {
60                  fprintf(stderr, "wave_out_open: alGetParamInfo failed: %s\n",                  fprintf(stderr, "sgi_open: alGetParamInfo failed: %s\n",
61                          alGetErrorString(oserror()));                          alGetErrorString(oserror()));
62          }          }
63          min_volume = alFixedToDouble(pinfo.min.ll);          min_volume = alFixedToDouble(pinfo.min.ll);
64          max_volume = alFixedToDouble(pinfo.max.ll);          max_volume = alFixedToDouble(pinfo.max.ll);
65          volume_range = (max_volume - min_volume);          volume_range = (max_volume - min_volume);
66  #if (defined(IRIX_DEBUG))  #if (defined(IRIX_DEBUG))
67          fprintf(stderr, "wave_out_open: minvol = %lf, maxvol= %lf, range = %lf.\n",          fprintf(stderr, "sgi_open: minvol = %lf, maxvol= %lf, range = %lf.\n",
68                  min_volume, max_volume, volume_range);                  min_volume, max_volume, volume_range);
69  #endif  #endif
70    
         queue_lo = queue_hi = 0;  
   
71          audioconfig = alNewConfig();          audioconfig = alNewConfig();
72          if (audioconfig < 0)          if (audioconfig == (ALconfig) 0)
73          {          {
74                  fprintf(stderr, "wave_out_open: alNewConfig failed: %s\n",                  fprintf(stderr, "sgi_open: alNewConfig failed: %s\n", alGetErrorString(oserror()));
                         alGetErrorString(oserror()));  
75                  return False;                  return False;
76          }          }
77    
78          output_port = alOpenPort("rdpsnd", "w", 0);          output_port = alOpenPort("rdpsnd", "w", 0);
79          if (output_port == (ALport) 0)          if (output_port == (ALport) 0)
80          {          {
81                  fprintf(stderr, "wave_out_open: alOpenPort failed: %s\n",                  fprintf(stderr, "sgi_open: alOpenPort failed: %s\n", alGetErrorString(oserror()));
                         alGetErrorString(oserror()));  
82                  return False;                  return False;
83          }          }
84    
85  #if (defined(IRIX_DEBUG))  #if (defined(IRIX_DEBUG))
86          fprintf(stderr, "wave_out_open: returning\n");          fprintf(stderr, "sgi_open: returning\n");
87  #endif  #endif
88          return True;          return True;
89  }  }
90    
91  void  void
92  wave_out_close(void)  sgi_close(void)
93  {  {
94          /* Ack all remaining packets */          /* Ack all remaining packets */
95  #if (defined(IRIX_DEBUG))  #if (defined(IRIX_DEBUG))
96          fprintf(stderr, "wave_out_close: begin\n");          fprintf(stderr, "sgi_close: begin\n");
97  #endif  #endif
98    
99          while (queue_lo != queue_hi)          while (!rdpsnd_queue_empty())
100          {                  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;  
         }  
101          alDiscardFrames(output_port, 0);          alDiscardFrames(output_port, 0);
102    
103          alClosePort(output_port);          alClosePort(output_port);
104          alFreeConfig(audioconfig);          alFreeConfig(audioconfig);
105  #if (defined(IRIX_DEBUG))  #if (defined(IRIX_DEBUG))
106          fprintf(stderr, "wave_out_close: returning\n");          fprintf(stderr, "sgi_close: returning\n");
107  #endif  #endif
108  }  }
109    
110  BOOL  BOOL
111  wave_out_format_supported(WAVEFORMATEX * pwfx)  sgi_format_supported(WAVEFORMATEX * pwfx)
112  {  {
113          if (pwfx->wFormatTag != WAVE_FORMAT_PCM)          if (pwfx->wFormatTag != WAVE_FORMAT_PCM)
114                  return False;                  return False;
# Line 136  wave_out_format_supported(WAVEFORMATEX * Line 121  wave_out_format_supported(WAVEFORMATEX *
121  }  }
122    
123  BOOL  BOOL
124  wave_out_set_format(WAVEFORMATEX * pwfx)  sgi_set_format(WAVEFORMATEX * pwfx)
125  {  {
126          int channels;          int channels;
127          int frameSize, channelCount;          int frameSize, channelCount;
128          ALpv params;          ALpv params;
129    
130  #if (defined(IRIX_DEBUG))  #if (defined(IRIX_DEBUG))
131          fprintf(stderr, "wave_out_set_format: init...\n");          fprintf(stderr, "sgi_set_format: init...\n");
132  #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;  
133    
134          if (pwfx->wBitsPerSample == 8)          if (pwfx->wBitsPerSample == 8)
135                  width = AL_SAMPLE_8;                  width = AL_SAMPLE_8;
136          else if (pwfx->wBitsPerSample == 16)          else if (pwfx->wBitsPerSample == 16)
         {  
137                  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  
         }  
138    
139          g_samplewidth = pwfx->wBitsPerSample / 8;          /* Limited support to configure an opened audio port in IRIX.  The
140               number of channels is a static setting and can not be changed after
141               a port is opened.  So if the number of channels remains the same, we
142               can configure other settings; otherwise we have to reopen the audio
143               port, using same config. */
144    
145          channels = pwfx->nChannels;          channels = pwfx->nChannels;
146          g_snd_rate = pwfx->nSamplesPerSec;          g_snd_rate = pwfx->nSamplesPerSec;
147    
148          alSetSampFmt(audioconfig, AL_SAMPFMT_TWOSCOMP);          alSetSampFmt(audioconfig, AL_SAMPFMT_TWOSCOMP);
149          alSetWidth(audioconfig, width);          alSetWidth(audioconfig, width);
150          alSetChannels(audioconfig, channels);          if (channels != alGetChannels(audioconfig))
151            {
152                    alClosePort(output_port);
153                    alSetChannels(audioconfig, channels);
154                    output_port = alOpenPort("rdpsnd", "w", audioconfig);
155    
156          output_port = alOpenPort("rdpsnd", "w", audioconfig);                  if (output_port == (ALport) 0)
157                    {
158                            fprintf(stderr, "sgi_set_format: alOpenPort failed: %s\n",
159                                    alGetErrorString(oserror()));
160                            return False;
161                    }
162    
         if (output_port == (ALport) 0)  
         {  
                 fprintf(stderr, "wave_out_set_format: alOpenPort failed: %s\n",  
                         alGetErrorString(oserror()));  
                 return False;  
163          }          }
164    
165          resource = alGetResource(output_port);          resource = alGetResource(output_port);
# Line 188  wave_out_set_format(WAVEFORMATEX * pwfx) Line 169  wave_out_set_format(WAVEFORMATEX * pwfx)
169    
170          if (frameSize == 0 || channelCount == 0)          if (frameSize == 0 || channelCount == 0)
171          {          {
172                  fprintf(stderr, "wave_out_set_format: bad frameSize or channelCount\n");                  fprintf(stderr, "sgi_set_format: bad frameSize or channelCount\n");
173                  return False;                  return False;
174          }          }
175          combinedFrameSize = frameSize * channelCount;          combinedFrameSize = frameSize * channelCount;
# Line 209  wave_out_set_format(WAVEFORMATEX * pwfx) Line 190  wave_out_set_format(WAVEFORMATEX * pwfx)
190          }          }
191    
192  #if (defined(IRIX_DEBUG))  #if (defined(IRIX_DEBUG))
193          fprintf(stderr, "wave_out_set_format: returning...\n");          fprintf(stderr, "sgi_set_format: returning...\n");
194  #endif  #endif
195          return True;          return True;
196  }  }
197    
198  void  void
199  wave_out_volume(uint16 left, uint16 right)  sgi_volume(uint16 left, uint16 right)
200  {  {
201          double gainleft, gainright;          double gainleft, gainright;
202          ALpv pv[1];          ALpv pv[1];
203          ALfixed gain[8];          ALfixed gain[8];
204    
205  #if (defined(IRIX_DEBUG))  #if (defined(IRIX_DEBUG))
206          fprintf(stderr, "wave_out_volume: begin\n");          fprintf(stderr, "sgi_volume: begin\n");
207          fprintf(stderr, "left='%d', right='%d'\n", left, right);          fprintf(stderr, "left='%d', right='%d'\n", left, right);
208  #endif  #endif
209    
# Line 237  wave_out_volume(uint16 left, uint16 righ Line 218  wave_out_volume(uint16 left, uint16 righ
218          pv[0].sizeIn = 8;          pv[0].sizeIn = 8;
219          if (alSetParams(AL_DEFAULT_OUTPUT, pv, 1) < 0)          if (alSetParams(AL_DEFAULT_OUTPUT, pv, 1) < 0)
220          {          {
221                  fprintf(stderr, "wave_out_volume: alSetParams failed: %s\n",                  fprintf(stderr, "sgi_volume: alSetParams failed: %s\n",
222                          alGetErrorString(oserror()));                          alGetErrorString(oserror()));
223                  return;                  return;
224          }          }
225    
226  #if (defined(IRIX_DEBUG))  #if (defined(IRIX_DEBUG))
227          fprintf(stderr, "wave_out_volume: returning\n");          fprintf(stderr, "sgi_volume: returning\n");
228  #endif  #endif
229  }  }
230    
231  void  void
232  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)  
233  {  {
234          struct audio_packet *packet;          struct audio_packet *packet;
235          ssize_t len;          ssize_t len;
236          unsigned int i;          unsigned int i;
         uint8 swap;  
237          STREAM out;          STREAM out;
         static long startedat_us;  
         static long startedat_s;  
         static BOOL started = False;  
         static BOOL swapped = False;  
         struct timeval tv;  
238          int gf;          int gf;
         static long long temp;  
239    
240          while (1)          while (1)
241          {          {
242                  if (queue_lo == queue_hi)                  if (rdpsnd_queue_empty())
243                  {                  {
244                          g_dsp_busy = False;                          g_dsp_busy = False;
245                          return;                          return;
246                  }                  }
247    
248                  packet = &packet_queue[queue_lo];                  packet = rdpsnd_queue_current_packet();
249                  out = &packet->s;                  out = &packet->s;
250    
                 /* 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;  
                 }  
   
251                  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  
                 }  
252    
253                  alWriteFrames(output_port, out->p, len / combinedFrameSize);                  alWriteFrames(output_port, out->p, len / combinedFrameSize);
254    
255                  out->p += len;                  out->p += len;
256                  if (out->p == out->end)                  if (out->p == out->end)
257                  {                  {
258                          long long duration;                          gf = alGetFilled(output_port);
259                          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)  
260                          {                          {
261                                  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;  
262                          }                          }
263                          else                          else
264                          {                          {
265  #if (defined(IRIX_DEBUG))  #if (defined(IRIX_DEBUG))
266                                  //fprintf(stderr,"Busy playing...\n");  /*                              fprintf(stderr,"Busy playing...\n"); */
267  #endif  #endif
268                                  g_dsp_busy = True;                                  g_dsp_busy = True;
269                                    usleep(10);
270                                  return;                                  return;
271                          }                          }
272                  }                  }
273          }          }
274  }  }
275    
276    static struct audio_driver sgi_driver = {
277            .name = "sgi",
278            .description = "SGI output driver",
279    
280            .wave_out_open = sgi_open,
281            .wave_out_close = sgi_close,
282            .wave_out_format_supported = sgi_format_supported,
283            .wave_out_set_format = sgi_set_format,
284            .wave_out_volume = sgi_volume,
285            .wave_out_play = sgi_play,
286    
287            .need_byteswap_on_be = 1,
288            .need_resampling = 0,
289    };
290    
291    struct audio_driver *
292    sgi_register(char *options)
293    {
294            if (options)
295            {
296                    sgi_output_device = xstrdup(options);
297            }
298            return &sgi_driver;
299    }

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

  ViewVC Help
Powered by ViewVC 1.1.26