2 Copyright (C) 2001 Paul Davis
3 Copyright (C) 2004 Grame
5 This program is free software; you can redistribute it and/or modify
6 it under the terms of the GNU General Public License as published by
7 the Free Software Foundation; either version 2 of the License, or
8 (at your option) any later version.
10 This program is distributed in the hope that it will be useful,
11 but WITHOUT ANY WARRANTY; without even the implied warranty of
12 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
13 GNU General Public License for more details.
15 You should have received a copy of the GNU General Public License
16 along with this program; if not, write to the Free Software
17 Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
21 #define __STDC_FORMAT_MACROS // For inttypes.h to work in C++
32 #include <sys/types.h>
36 #include "JackAlsaDriver.h"
37 #include "JackEngineControl.h"
38 #include "JackClientControl.h"
40 #include "JackGraphManager.h"
41 #include "JackLockedEngine.h"
43 #include "JackAndroidThread.h"
45 #include "JackPosixThread.h"
47 #include "JackCompilerDeps.h"
48 #include "JackServerGlobals.h"
50 static struct jack_constraint_enum_str_descriptor midi_constraint_descr_array
[] =
52 { "none", "no MIDI driver" },
53 { "seq", "ALSA Sequencer driver" },
54 { "raw", "ALSA RawMIDI driver" },
58 static struct jack_constraint_enum_char_descriptor dither_constraint_descr_array
[] =
61 { 'r', "rectangular" },
63 { 't', "triangular" },
70 static volatile bool device_reservation_loop_running
= false;
72 static void* on_device_reservation_loop(void*)
74 while (device_reservation_loop_running
&& JackServerGlobals::on_device_reservation_loop
!= NULL
) {
75 JackServerGlobals::on_device_reservation_loop();
82 int JackAlsaDriver::SetBufferSize(jack_nframes_t buffer_size
)
84 jack_log("JackAlsaDriver::SetBufferSize %ld", buffer_size
);
85 int res
= alsa_driver_reset_parameters((alsa_driver_t
*)fDriver
, buffer_size
,
86 ((alsa_driver_t
*)fDriver
)->user_nperiods
,
87 ((alsa_driver_t
*)fDriver
)->frame_rate
);
89 if (res
== 0) { // update fEngineControl and fGraphManager
90 JackAudioDriver::SetBufferSize(buffer_size
); // Generic change, never fails
95 alsa_driver_reset_parameters((alsa_driver_t
*)fDriver
, fEngineControl
->fBufferSize
,
96 ((alsa_driver_t
*)fDriver
)->user_nperiods
,
97 ((alsa_driver_t
*)fDriver
)->frame_rate
);
103 void JackAlsaDriver::UpdateLatencies()
105 jack_latency_range_t range
;
106 alsa_driver_t
* alsa_driver
= (alsa_driver_t
*)fDriver
;
108 for (int i
= 0; i
< fCaptureChannels
; i
++) {
109 range
.min
= range
.max
= alsa_driver
->frames_per_cycle
+ alsa_driver
->capture_frame_latency
;
110 fGraphManager
->GetPort(fCapturePortList
[i
])->SetLatencyRange(JackCaptureLatency
, &range
);
113 for (int i
= 0; i
< fPlaybackChannels
; i
++) {
114 // Add one buffer more latency if "async" mode is used...
115 range
.min
= range
.max
= (alsa_driver
->frames_per_cycle
* (alsa_driver
->user_nperiods
- 1)) +
116 ((fEngineControl
->fSyncMode
) ? 0 : fEngineControl
->fBufferSize
) + alsa_driver
->playback_frame_latency
;
117 fGraphManager
->GetPort(fPlaybackPortList
[i
])->SetLatencyRange(JackPlaybackLatency
, &range
);
119 if (fWithMonitorPorts
) {
120 range
.min
= range
.max
= alsa_driver
->frames_per_cycle
;
121 fGraphManager
->GetPort(fMonitorPortList
[i
])->SetLatencyRange(JackCaptureLatency
, &range
);
126 int JackAlsaDriver::Attach()
129 jack_port_id_t port_index
;
130 unsigned long port_flags
= (unsigned long)CaptureDriverFlags
;
131 char name
[REAL_JACK_PORT_NAME_SIZE
+1];
132 char alias
[REAL_JACK_PORT_NAME_SIZE
+1];
134 assert(fCaptureChannels
< DRIVER_PORT_NUM
);
135 assert(fPlaybackChannels
< DRIVER_PORT_NUM
);
137 alsa_driver_t
* alsa_driver
= (alsa_driver_t
*)fDriver
;
139 if (alsa_driver
->has_hw_monitoring
)
140 port_flags
|= JackPortCanMonitor
;
142 // ALSA driver may have changed the values
143 JackAudioDriver::SetBufferSize(alsa_driver
->frames_per_cycle
);
144 JackAudioDriver::SetSampleRate(alsa_driver
->frame_rate
);
146 jack_log("JackAlsaDriver::Attach fBufferSize %ld fSampleRate %ld", fEngineControl
->fBufferSize
, fEngineControl
->fSampleRate
);
148 for (int i
= 0; i
< fCaptureChannels
; i
++) {
149 snprintf(alias
, sizeof(alias
), "%s:%s:out%d", fAliasName
, fCaptureDriverName
, i
+ 1);
150 snprintf(name
, sizeof(name
), "%s:capture_%d", fClientControl
.fName
, i
+ 1);
151 if (fEngine
->PortRegister(fClientControl
.fRefNum
, name
, JACK_DEFAULT_AUDIO_TYPE
, (JackPortFlags
)port_flags
, fEngineControl
->fBufferSize
, &port_index
) < 0) {
152 jack_error("driver: cannot register port for %s", name
);
155 port
= fGraphManager
->GetPort(port_index
);
156 port
->SetAlias(alias
);
157 fCapturePortList
[i
] = port_index
;
158 jack_log("JackAlsaDriver::Attach fCapturePortList[i] %ld ", port_index
);
161 port_flags
= (unsigned long)PlaybackDriverFlags
;
163 for (int i
= 0; i
< fPlaybackChannels
; i
++) {
164 snprintf(alias
, sizeof(alias
), "%s:%s:in%d", fAliasName
, fPlaybackDriverName
, i
+ 1);
165 snprintf(name
, sizeof(name
), "%s:playback_%d", fClientControl
.fName
, i
+ 1);
166 if (fEngine
->PortRegister(fClientControl
.fRefNum
, name
, JACK_DEFAULT_AUDIO_TYPE
, (JackPortFlags
)port_flags
, fEngineControl
->fBufferSize
, &port_index
) < 0) {
167 jack_error("driver: cannot register port for %s", name
);
170 port
= fGraphManager
->GetPort(port_index
);
171 port
->SetAlias(alias
);
172 fPlaybackPortList
[i
] = port_index
;
173 jack_log("JackAlsaDriver::Attach fPlaybackPortList[i] %ld ", port_index
);
176 if (fWithMonitorPorts
) {
177 jack_log("Create monitor port");
178 snprintf(name
, sizeof(name
), "%s:monitor_%d", fClientControl
.fName
, i
+ 1);
179 if (fEngine
->PortRegister(fClientControl
.fRefNum
, name
, JACK_DEFAULT_AUDIO_TYPE
, MonitorDriverFlags
, fEngineControl
->fBufferSize
, &port_index
) < 0) {
180 jack_error("ALSA: cannot register monitor port for %s", name
);
182 fMonitorPortList
[i
] = port_index
;
189 if (alsa_driver
->midi
) {
190 int err
= (alsa_driver
->midi
->attach
)(alsa_driver
->midi
);
192 jack_error ("ALSA: cannot attach MIDI: %d", err
);
198 int JackAlsaDriver::Detach()
200 alsa_driver_t
* alsa_driver
= (alsa_driver_t
*)fDriver
;
201 if (alsa_driver
->midi
)
202 (alsa_driver
->midi
->detach
)(alsa_driver
->midi
);
204 return JackAudioDriver::Detach();
207 extern "C" char* get_control_device_name(const char * device_name
)
212 /* the user wants a hw or plughw device, the ctl name
213 * should be hw:x where x is the card identification.
214 * We skip the subdevice suffix that starts with comma */
216 if (strncasecmp(device_name
, "plughw:", 7) == 0) {
217 /* skip the "plug" prefix" */
221 comma
= strchr(device_name
, ',');
223 ctl_name
= strdup(device_name
);
224 if (ctl_name
== NULL
) {
225 jack_error("strdup(\"%s\") failed.", device_name
);
228 ctl_name
= strndup(device_name
, comma
- device_name
);
229 if (ctl_name
== NULL
) {
230 jack_error("strndup(\"%s\", %u) failed.", device_name
, (unsigned int)(comma
- device_name
));
237 static int card_to_num(const char* device
)
241 snd_ctl_card_info_t
*card_info
;
242 snd_ctl_t
* ctl_handle
;
245 snd_ctl_card_info_alloca (&card_info
);
247 ctl_name
= get_control_device_name(device
);
248 if (ctl_name
== NULL
) {
249 jack_error("get_control_device_name() failed.");
253 if ((err
= snd_ctl_open (&ctl_handle
, ctl_name
, 0)) < 0) {
254 jack_error ("control open \"%s\" (%s)", ctl_name
,
259 if ((err
= snd_ctl_card_info(ctl_handle
, card_info
)) < 0) {
260 jack_error ("control hardware info \"%s\" (%s)",
261 device
, snd_strerror (err
));
265 i
= snd_ctl_card_info_get_card(card_info
);
268 snd_ctl_close(ctl_handle
);
277 int JackAlsaDriver::Open(jack_nframes_t nframes
,
278 jack_nframes_t user_nperiods
,
279 jack_nframes_t samplerate
,
284 DitherAlgorithm dither
,
290 const char* capture_driver_name
,
291 const char* playback_driver_name
,
292 jack_nframes_t capture_latency
,
293 jack_nframes_t playback_latency
,
294 const char* midi_driver_name
)
296 // Generic JackAudioDriver Open
297 if (JackAudioDriver::Open(nframes
, samplerate
, capturing
, playing
,
298 inchannels
, outchannels
, monitor
, capture_driver_name
, playback_driver_name
,
299 capture_latency
, playback_latency
) != 0) {
303 alsa_midi_t
*midi
= 0;
305 if (strcmp(midi_driver_name
, "seq") == 0)
306 midi
= alsa_seqmidi_new((jack_client_t
*)this, 0);
307 else if (strcmp(midi_driver_name
, "raw") == 0)
308 midi
= alsa_rawmidi_new((jack_client_t
*)this);
311 if (JackServerGlobals::on_device_acquire
!= NULL
) {
312 int capture_card
= card_to_num(capture_driver_name
);
313 int playback_card
= card_to_num(playback_driver_name
);
316 if (capture_card
>= 0) {
317 snprintf(audio_name
, sizeof(audio_name
), "Audio%d", capture_card
);
318 if (!JackServerGlobals::on_device_acquire(audio_name
)) {
319 jack_error("Audio device %s cannot be acquired...", capture_driver_name
);
324 if (playback_card
>= 0 && playback_card
!= capture_card
) {
325 snprintf(audio_name
, sizeof(audio_name
), "Audio%d", playback_card
);
326 if (!JackServerGlobals::on_device_acquire(audio_name
)) {
327 jack_error("Audio device %s cannot be acquired...", playback_driver_name
);
328 if (capture_card
>= 0) {
329 snprintf(audio_name
, sizeof(audio_name
), "Audio%d", capture_card
);
330 JackServerGlobals::on_device_release(audio_name
);
337 fDriver
= alsa_driver_new ((char*)"alsa_pcm", (char*)playback_driver_name
, (char*)capture_driver_name
,
360 // ALSA driver may have changed the in/out values
361 fCaptureChannels
= ((alsa_driver_t
*)fDriver
)->capture_nchannels
;
362 fPlaybackChannels
= ((alsa_driver_t
*)fDriver
)->playback_nchannels
;
363 if (JackServerGlobals::on_device_reservation_loop
!= NULL
) {
364 device_reservation_loop_running
= true;
365 if (JackPosixThread::StartImp(&fReservationLoopThread
, 0, 0, on_device_reservation_loop
, NULL
) != 0) {
366 device_reservation_loop_running
= false;
373 int JackAlsaDriver::Close()
375 // Generic audio driver close
376 int res
= JackAudioDriver::Close();
379 alsa_driver_delete((alsa_driver_t
*)fDriver
);
382 if (device_reservation_loop_running
) {
383 device_reservation_loop_running
= false;
384 JackPosixThread::StopImp(fReservationLoopThread
);
387 if (JackServerGlobals::on_device_release
!= NULL
)
390 int capture_card
= card_to_num(fCaptureDriverName
);
391 if (capture_card
>= 0) {
392 snprintf(audio_name
, sizeof(audio_name
), "Audio%d", capture_card
);
393 JackServerGlobals::on_device_release(audio_name
);
396 int playback_card
= card_to_num(fPlaybackDriverName
);
397 if (playback_card
>= 0 && playback_card
!= capture_card
) {
398 snprintf(audio_name
, sizeof(audio_name
), "Audio%d", playback_card
);
399 JackServerGlobals::on_device_release(audio_name
);
406 int JackAlsaDriver::Start()
408 int res
= JackAudioDriver::Start();
410 res
= alsa_driver_start((alsa_driver_t
*)fDriver
);
412 JackAudioDriver::Stop();
418 int JackAlsaDriver::Stop()
420 int res
= alsa_driver_stop((alsa_driver_t
*)fDriver
);
421 if (JackAudioDriver::Stop() < 0) {
427 int JackAlsaDriver::Read()
429 /* Taken from alsa_driver_run_cycle */
431 jack_nframes_t nframes
;
436 nframes
= alsa_driver_wait((alsa_driver_t
*)fDriver
, -1, &wait_status
, &fDelayedUsecs
);
439 return -1; /* driver failed */
442 /* we detected an xrun and restarted: notify
443 * clients about the delay.
445 jack_log("ALSA XRun wait_status = %d", wait_status
);
446 NotifyXRun(fBeginDateUst
, fDelayedUsecs
);
447 goto retry
; /* recoverable error*/
450 if (nframes
!= fEngineControl
->fBufferSize
)
451 jack_log("JackAlsaDriver::Read warning fBufferSize = %ld nframes = %ld", fEngineControl
->fBufferSize
, nframes
);
453 // Has to be done before read
454 JackDriver::CycleIncTime();
456 return alsa_driver_read((alsa_driver_t
*)fDriver
, fEngineControl
->fBufferSize
);
459 int JackAlsaDriver::Write()
461 return alsa_driver_write((alsa_driver_t
*)fDriver
, fEngineControl
->fBufferSize
);
464 void JackAlsaDriver::ReadInputAux(jack_nframes_t orig_nframes
, snd_pcm_sframes_t contiguous
, snd_pcm_sframes_t nread
)
466 for (int chn
= 0; chn
< fCaptureChannels
; chn
++) {
467 if (fGraphManager
->GetConnectionsNum(fCapturePortList
[chn
]) > 0) {
468 jack_default_audio_sample_t
* buf
= (jack_default_audio_sample_t
*)fGraphManager
->GetBuffer(fCapturePortList
[chn
], orig_nframes
);
469 alsa_driver_read_from_channel((alsa_driver_t
*)fDriver
, chn
, buf
+ nread
, contiguous
);
474 void JackAlsaDriver::MonitorInputAux()
476 for (int chn
= 0; chn
< fCaptureChannels
; chn
++) {
477 JackPort
* port
= fGraphManager
->GetPort(fCapturePortList
[chn
]);
478 if (port
->MonitoringInput()) {
479 ((alsa_driver_t
*)fDriver
)->input_monitor_mask
|= (1 << chn
);
484 void JackAlsaDriver::ClearOutputAux()
486 for (int chn
= 0; chn
< fPlaybackChannels
; chn
++) {
487 jack_default_audio_sample_t
* buf
=
488 (jack_default_audio_sample_t
*)fGraphManager
->GetBuffer(fPlaybackPortList
[chn
], fEngineControl
->fBufferSize
);
489 memset(buf
, 0, sizeof (jack_default_audio_sample_t
) * fEngineControl
->fBufferSize
);
493 void JackAlsaDriver::SetTimetAux(jack_time_t time
)
495 fBeginDateUst
= time
;
498 int JackAlsaDriver::PortSetDefaultMetadata(jack_port_id_t port_id
, const char* pretty_name
)
500 return fEngine
->PortSetDefaultMetadata(fClientControl
.fRefNum
, port_id
, pretty_name
);
503 void JackAlsaDriver::WriteOutputAux(jack_nframes_t orig_nframes
, snd_pcm_sframes_t contiguous
, snd_pcm_sframes_t nwritten
)
505 for (int chn
= 0; chn
< fPlaybackChannels
; chn
++) {
507 if (fGraphManager
->GetConnectionsNum(fPlaybackPortList
[chn
]) > 0) {
508 jack_default_audio_sample_t
* buf
= (jack_default_audio_sample_t
*)fGraphManager
->GetBuffer(fPlaybackPortList
[chn
], orig_nframes
);
509 alsa_driver_write_to_channel(((alsa_driver_t
*)fDriver
), chn
, buf
+ nwritten
, contiguous
);
511 if (fWithMonitorPorts
&& fGraphManager
->GetConnectionsNum(fMonitorPortList
[chn
]) > 0) {
512 jack_default_audio_sample_t
* monbuf
= (jack_default_audio_sample_t
*)fGraphManager
->GetBuffer(fMonitorPortList
[chn
], orig_nframes
);
513 memcpy(monbuf
+ nwritten
, buf
+ nwritten
, contiguous
* sizeof(jack_default_audio_sample_t
));
519 int JackAlsaDriver::is_realtime() const
521 return fEngineControl
->fRealTime
;
524 int JackAlsaDriver::create_thread(pthread_t
*thread
, int priority
, int realtime
, void *(*start_routine
)(void*), void *arg
)
527 return JackAndroidThread::StartImp(thread
, priority
, realtime
, start_routine
, arg
);
529 return JackPosixThread::StartImp(thread
, priority
, realtime
, start_routine
, arg
);
533 jack_port_id_t
JackAlsaDriver::port_register(const char *port_name
, const char *port_type
, unsigned long flags
, unsigned long buffer_size
)
535 jack_port_id_t port_index
;
536 int res
= fEngine
->PortRegister(fClientControl
.fRefNum
, port_name
, port_type
, flags
, buffer_size
, &port_index
);
537 return (res
== 0) ? port_index
: 0;
540 int JackAlsaDriver::port_unregister(jack_port_id_t port_index
)
542 return fEngine
->PortUnRegister(fClientControl
.fRefNum
, port_index
);
545 void* JackAlsaDriver::port_get_buffer(int port
, jack_nframes_t nframes
)
547 return fGraphManager
->GetBuffer(port
, nframes
);
550 int JackAlsaDriver::port_set_alias(int port
, const char* name
)
552 return fGraphManager
->GetPort(port
)->SetAlias(name
);
555 jack_nframes_t
JackAlsaDriver::get_sample_rate() const
557 return fEngineControl
->fSampleRate
;
560 jack_nframes_t
JackAlsaDriver::frame_time() const
563 fEngineControl
->ReadFrameTime(&timer
);
564 return timer
.Time2Frames(GetMicroSeconds(), fEngineControl
->fBufferSize
);
567 jack_nframes_t
JackAlsaDriver::last_frame_time() const
570 fEngineControl
->ReadFrameTime(&timer
);
571 return timer
.CurFrame();
574 } // end of namespace
583 jack_driver_param_constraint_desc_t
*
587 snd_ctl_card_info_t
* info
;
588 snd_pcm_info_t
* pcminfo_capture
;
589 snd_pcm_info_t
* pcminfo_playback
;
591 jack_driver_param_value_t card_id
;
592 jack_driver_param_value_t device_id
;
593 char description
[64];
597 jack_driver_param_constraint_desc_t
* constraint_ptr
;
598 uint32_t array_size
= 0;
600 snd_ctl_card_info_alloca(&info
);
601 snd_pcm_info_alloca(&pcminfo_capture
);
602 snd_pcm_info_alloca(&pcminfo_playback
);
604 constraint_ptr
= NULL
;
606 while(snd_card_next(&card_no
) >= 0 && card_no
>= 0)
608 snprintf(card_id
.str
, sizeof(card_id
.str
), "hw:%d", card_no
);
610 if (snd_ctl_open(&handle
, card_id
.str
, 0) >= 0 &&
611 snd_ctl_card_info(handle
, info
) >= 0)
613 snprintf(card_id
.str
, sizeof(card_id
.str
), "hw:%s", snd_ctl_card_info_get_id(info
));
614 if (!jack_constraint_add_enum(
618 snd_ctl_card_info_get_name(info
)))
623 while (snd_ctl_pcm_next_device(handle
, &device_no
) >= 0 && device_no
!= -1)
625 snprintf(device_id
.str
, sizeof(device_id
.str
), "%s,%d", card_id
.str
, device_no
);
627 snd_pcm_info_set_device(pcminfo_capture
, device_no
);
628 snd_pcm_info_set_subdevice(pcminfo_capture
, 0);
629 snd_pcm_info_set_stream(pcminfo_capture
, SND_PCM_STREAM_CAPTURE
);
630 has_capture
= snd_ctl_pcm_info(handle
, pcminfo_capture
) >= 0;
632 snd_pcm_info_set_device(pcminfo_playback
, device_no
);
633 snd_pcm_info_set_subdevice(pcminfo_playback
, 0);
634 snd_pcm_info_set_stream(pcminfo_playback
, SND_PCM_STREAM_PLAYBACK
);
635 has_playback
= snd_ctl_pcm_info(handle
, pcminfo_playback
) >= 0;
637 if (has_capture
&& has_playback
)
639 snprintf(description
, sizeof(description
),"%s (duplex)", snd_pcm_info_get_name(pcminfo_capture
));
641 else if (has_capture
)
643 snprintf(description
, sizeof(description
),"%s (capture)", snd_pcm_info_get_name(pcminfo_capture
));
645 else if (has_playback
)
647 snprintf(description
, sizeof(description
),"%s (playback)", snd_pcm_info_get_name(pcminfo_playback
));
654 if (!jack_constraint_add_enum(
662 snd_ctl_close(handle
);
666 return constraint_ptr
;
668 jack_constraint_free(constraint_ptr
);
673 dither_opt (char c
, DitherAlgorithm
* dither
)
682 *dither
= Rectangular
;
690 *dither
= Triangular
;
694 fprintf (stderr
, "ALSA driver: illegal dithering mode %c\n", c
);
700 SERVER_EXPORT
const jack_driver_desc_t
* driver_get_descriptor ()
702 jack_driver_desc_t
* desc
;
703 jack_driver_desc_filler_t filler
;
704 jack_driver_param_value_t value
;
706 desc
= jack_driver_descriptor_construct("alsa", JackDriverMaster
, "Linux ALSA API based audio backend", &filler
);
708 strcpy(value
.str
, "hw:0");
710 jack_driver_descriptor_add_parameter(desc
, &filler
, "device", 'd', JackDriverParamString
, &value
, NULL
, "ALSA device name", NULL
);
712 jack_driver_descriptor_add_parameter(desc
, &filler
, "device", 'd', JackDriverParamString
, &value
, enum_alsa_devices(), "ALSA device name", NULL
);
715 strcpy(value
.str
, "none");
716 jack_driver_descriptor_add_parameter(desc
, &filler
, "capture", 'C', JackDriverParamString
, &value
, NULL
, "Provide capture ports. Optionally set device", NULL
);
717 jack_driver_descriptor_add_parameter(desc
, &filler
, "playback", 'P', JackDriverParamString
, &value
, NULL
, "Provide playback ports. Optionally set device", NULL
);
720 jack_driver_descriptor_add_parameter(desc
, &filler
, "rate", 'r', JackDriverParamUInt
, &value
, NULL
, "Sample rate", NULL
);
723 jack_driver_descriptor_add_parameter(desc
, &filler
, "period", 'p', JackDriverParamUInt
, &value
, NULL
, "Frames per period", NULL
);
726 jack_driver_descriptor_add_parameter(desc
, &filler
, "nperiods", 'n', JackDriverParamUInt
, &value
, NULL
, "Number of periods of playback latency", NULL
);
729 jack_driver_descriptor_add_parameter(desc
, &filler
, "hwmon", 'H', JackDriverParamBool
, &value
, NULL
, "Hardware monitoring, if available", NULL
);
732 jack_driver_descriptor_add_parameter(desc
, &filler
, "hwmeter", 'M', JackDriverParamBool
, &value
, NULL
, "Hardware metering, if available", NULL
);
735 jack_driver_descriptor_add_parameter(desc
, &filler
, "duplex", 'D', JackDriverParamBool
, &value
, NULL
, "Provide both capture and playback ports", NULL
);
738 jack_driver_descriptor_add_parameter(desc
, &filler
, "softmode", 's', JackDriverParamBool
, &value
, NULL
, "Soft-mode, no xrun handling", NULL
);
741 jack_driver_descriptor_add_parameter(desc
, &filler
, "monitor", 'm', JackDriverParamBool
, &value
, NULL
, "Provide monitor ports for the output", NULL
);
744 jack_driver_descriptor_add_parameter(
751 jack_constraint_compose_enum_char(
752 JACK_CONSTRAINT_FLAG_STRICT
| JACK_CONSTRAINT_FLAG_FAKE_VALUE
,
753 dither_constraint_descr_array
),
758 jack_driver_descriptor_add_parameter(desc
, &filler
, "inchannels", 'i', JackDriverParamUInt
, &value
, NULL
, "Number of capture channels (defaults to hardware max)", NULL
);
759 jack_driver_descriptor_add_parameter(desc
, &filler
, "outchannels", 'o', JackDriverParamUInt
, &value
, NULL
, "Number of playback channels (defaults to hardware max)", NULL
);
762 jack_driver_descriptor_add_parameter(desc
, &filler
, "shorts", 'S', JackDriverParamBool
, &value
, NULL
, "Try 16-bit samples before 32-bit", NULL
);
765 jack_driver_descriptor_add_parameter(desc
, &filler
, "input-latency", 'I', JackDriverParamUInt
, &value
, NULL
, "Extra input latency (frames)", NULL
);
766 jack_driver_descriptor_add_parameter(desc
, &filler
, "output-latency", 'O', JackDriverParamUInt
, &value
, NULL
, "Extra output latency (frames)", NULL
);
768 strcpy(value
.str
, "none");
769 jack_driver_descriptor_add_parameter(
774 JackDriverParamString
,
776 jack_constraint_compose_enum_str(
777 JACK_CONSTRAINT_FLAG_STRICT
| JACK_CONSTRAINT_FLAG_FAKE_VALUE
,
778 midi_constraint_descr_array
),
785 static Jack::JackAlsaDriver
* g_alsa_driver
;
787 SERVER_EXPORT
Jack::JackDriverClientInterface
* driver_initialize(Jack::JackLockedEngine
* engine
, Jack::JackSynchro
* table
, const JSList
* params
)
789 jack_nframes_t srate
= 48000;
790 jack_nframes_t frames_per_interrupt
= 1024;
791 unsigned long user_nperiods
= 2;
792 const char *playback_pcm_name
= "hw:0";
793 const char *capture_pcm_name
= "hw:0";
794 int hw_monitoring
= FALSE
;
795 int hw_metering
= FALSE
;
797 int playback
= FALSE
;
798 int soft_mode
= FALSE
;
800 DitherAlgorithm dither
= None
;
801 int user_capture_nchnls
= 0;
802 int user_playback_nchnls
= 0;
803 int shorts_first
= FALSE
;
804 jack_nframes_t systemic_input_latency
= 0;
805 jack_nframes_t systemic_output_latency
= 0;
807 const jack_driver_param_t
* param
;
808 const char *midi_driver
= "none";
810 for (node
= params
; node
; node
= jack_slist_next (node
)) {
811 param
= (const jack_driver_param_t
*) node
->data
;
813 switch (param
->character
) {
817 if (strcmp (param
->value
.str
, "none") != 0) {
818 capture_pcm_name
= strdup (param
->value
.str
);
819 jack_log("capture device %s", capture_pcm_name
);
825 if (strcmp (param
->value
.str
, "none") != 0) {
826 playback_pcm_name
= strdup (param
->value
.str
);
827 jack_log("playback device %s", playback_pcm_name
);
837 if (strcmp (param
->value
.str
, "none") != 0) {
838 playback_pcm_name
= strdup (param
->value
.str
);
839 capture_pcm_name
= strdup (param
->value
.str
);
840 jack_log("playback device %s", playback_pcm_name
);
841 jack_log("capture device %s", capture_pcm_name
);
846 hw_monitoring
= param
->value
.i
;
850 monitor
= param
->value
.i
;
854 hw_metering
= param
->value
.i
;
858 srate
= param
->value
.ui
;
859 jack_log("apparent rate = %d", srate
);
863 frames_per_interrupt
= param
->value
.ui
;
864 jack_log("frames per period = %d", frames_per_interrupt
);
868 user_nperiods
= param
->value
.ui
;
869 if (user_nperiods
< 2) { /* enforce minimum value */
875 soft_mode
= param
->value
.i
;
879 if (dither_opt (param
->value
.c
, &dither
)) {
885 user_capture_nchnls
= param
->value
.ui
;
889 user_playback_nchnls
= param
->value
.ui
;
893 shorts_first
= param
->value
.i
;
897 systemic_input_latency
= param
->value
.ui
;
901 systemic_output_latency
= param
->value
.ui
;
905 midi_driver
= strdup(param
->value
.str
);
910 /* duplex is the default */
911 if (!capture
&& !playback
) {
916 g_alsa_driver
= new Jack::JackAlsaDriver("system", "alsa_pcm", engine
, table
);
917 Jack::JackDriverClientInterface
* threaded_driver
= new Jack::JackThreadedDriver(g_alsa_driver
);
918 // Special open for ALSA driver...
919 if (g_alsa_driver
->Open(frames_per_interrupt
, user_nperiods
, srate
, hw_monitoring
, hw_metering
, capture
, playback
, dither
, soft_mode
, monitor
,
920 user_capture_nchnls
, user_playback_nchnls
, shorts_first
, capture_pcm_name
, playback_pcm_name
,
921 systemic_input_latency
, systemic_output_latency
, midi_driver
) == 0) {
922 return threaded_driver
;
924 delete threaded_driver
; // Delete the decorated driver
929 // Code to be used in alsa_driver.c
931 void ReadInput(jack_nframes_t orig_nframes
, snd_pcm_sframes_t contiguous
, snd_pcm_sframes_t nread
)
933 g_alsa_driver
->ReadInputAux(orig_nframes
, contiguous
, nread
);
937 g_alsa_driver
->MonitorInputAux();
941 g_alsa_driver
->ClearOutputAux();
943 void WriteOutput(jack_nframes_t orig_nframes
, snd_pcm_sframes_t contiguous
, snd_pcm_sframes_t nwritten
)
945 g_alsa_driver
->WriteOutputAux(orig_nframes
, contiguous
, nwritten
);
947 void SetTime(jack_time_t time
)
949 g_alsa_driver
->SetTimetAux(time
);
955 if ((res
= g_alsa_driver
->Stop()) == 0) {
956 res
= g_alsa_driver
->Start();