|
- /*
- Copyright (C) 2002 Anthony Van Groningen
-
- Parts based on source code taken from the
- "Env24 chipset (ICE1712) control utility" that is
-
- Copyright (C) 2000 by Jaroslav Kysela <perex@suse.cz>
-
- This program is free software; you can redistribute it and/or modify
- it under the terms of the GNU General Public License as published by
- the Free Software Foundation; either version 2 of the License, or
- (at your option) any later version.
-
- This program is distributed in the hope that it will be useful,
- but WITHOUT ANY WARRANTY; without even the implied warranty of
- MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
- GNU General Public License for more details.
-
- You should have received a copy of the GNU General Public License
- along with this program; if not, write to the Free Software
- Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
- */
-
- #if defined(HAVE_CONFIG_H)
- #include "config.h"
- #endif
-
- //#include <jack/hardware.h>
-
- #include "hardware.h"
- #include "alsa_driver.h"
- #include "ice1712.h"
- #include "JackError.h"
-
- static int
- ice1712_hw_monitor_toggle(jack_hardware_t *hw, int idx, int onoff)
- {
- ice1712_t *h = (ice1712_t *) hw->private_hw;
- snd_ctl_elem_value_t *val;
- int err;
-
- snd_ctl_elem_value_alloca (&val);
- snd_ctl_elem_value_set_interface (val, SND_CTL_ELEM_IFACE_MIXER);
- if (idx >= 8) {
- snd_ctl_elem_value_set_name (val, SPDIF_PLAYBACK_ROUTE_NAME);
- snd_ctl_elem_value_set_index (val, idx - 8);
- } else {
- snd_ctl_elem_value_set_name (val, ANALOG_PLAYBACK_ROUTE_NAME);
- snd_ctl_elem_value_set_index (val, idx);
- }
- if (onoff) {
- snd_ctl_elem_value_set_enumerated (val, 0, idx + 1);
- } else {
- snd_ctl_elem_value_set_enumerated (val, 0, 0);
- }
- if ((err = snd_ctl_elem_write (h->driver->ctl_handle, val)) != 0) {
- jack_error ("ALSA/ICE1712: (%d) cannot set input monitoring (%s)",
- idx,snd_strerror (err));
- return -1;
- }
-
- return 0;
- }
-
- static int
- ice1712_set_input_monitor_mask (jack_hardware_t *hw, unsigned long mask)
- {
- int idx;
- ice1712_t *h = (ice1712_t *) hw->private_hw;
-
- for (idx = 0; idx < 10; idx++) {
- if (h->active_channels & (1<<idx)) {
- ice1712_hw_monitor_toggle (hw, idx, mask & (1<<idx) ? 1 : 0);
- }
- }
- hw->input_monitor_mask = mask;
-
- return 0;
- }
-
- static int
- ice1712_change_sample_clock (jack_hardware_t *hw, SampleClockMode mode)
- {
- return -1;
- }
-
- static void
- ice1712_release (jack_hardware_t *hw)
- {
- ice1712_t *h = (ice1712_t *) hw->private_hw;
-
- if (h == 0)
- return;
-
- if (h->eeprom)
- free(h->eeprom);
-
- free(h);
- }
-
-
- jack_hardware_t *
- jack_alsa_ice1712_hw_new (alsa_driver_t *driver)
- {
- jack_hardware_t *hw;
- ice1712_t *h;
- snd_ctl_elem_value_t *val;
- int err;
-
- hw = (jack_hardware_t *) malloc (sizeof (jack_hardware_t));
-
- hw->capabilities = Cap_HardwareMonitoring;
- hw->input_monitor_mask = 0;
- hw->private_hw = 0;
-
- hw->set_input_monitor_mask = ice1712_set_input_monitor_mask;
- hw->change_sample_clock = ice1712_change_sample_clock;
- hw->release = ice1712_release;
-
- h = (ice1712_t *) malloc (sizeof (ice1712_t));
-
- h->driver = driver;
-
- /* Get the EEPROM (adopted from envy24control) */
- h->eeprom = (ice1712_eeprom_t *) malloc (sizeof (ice1712_eeprom_t));
- snd_ctl_elem_value_alloca (&val);
- snd_ctl_elem_value_set_interface (val, SND_CTL_ELEM_IFACE_CARD);
- snd_ctl_elem_value_set_name (val, "ICE1712 EEPROM");
- if ((err = snd_ctl_elem_read (driver->ctl_handle, val)) < 0) {
- jack_error( "ALSA/ICE1712: Unable to read EEPROM contents (%s)\n", snd_strerror (err));
- /* Recover? */
- }
- memcpy(h->eeprom, snd_ctl_elem_value_get_bytes(val), 32);
-
- /* determine number of pro ADC's. We're asumming that there is at least one stereo pair.
- Should check this first, but how? */
- switch((h->eeprom->codec & 0xCU) >> 2) {
- case 0:
- h->active_channels = 0x3U;
- break;
- case 1:
- h->active_channels = 0xfU;
- break;
- case 2:
- h->active_channels = 0x3fU;
- break;
- case 3:
- h->active_channels = 0xffU;
- break;
- }
- /* check for SPDIF In's */
- if (h->eeprom->spdif & 0x1U) {
- h->active_channels |= 0x300U;
- }
-
- hw->private_hw = h;
-
- return hw;
- }
|