/***************************************************************************** hardware.c - Hardware Settings Copyright (C) 2000 by Jaroslav Kysela 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., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA. ******************************************************************************/ #include "envy24control.h" static snd_ctl_elem_t spdif_master; static snd_ctl_elem_t word_clock_sync; static snd_ctl_elem_t volume_rate; static snd_ctl_elem_t spdif_input; static snd_ctl_elem_t spdif_output; #define toggle_set(widget, state) \ gtk_toggle_button_set_active(GTK_TOGGLE_BUTTON(widget), state); static int is_active(GtkWidget *widget) { return GTK_TOGGLE_BUTTON(widget)->active ? 1 : 0; } void master_clock_update(void) { int err; if ((err = snd_ctl_elem_read(card_ctl, &spdif_master)) < 0) g_print("Unable to read S/PDIF master state: %s\n", snd_strerror(err)); if (card_eeprom.subvendor == ICE1712_SUBDEVICE_DELTA1010) { if ((err = snd_ctl_elem_read(card_ctl, &word_clock_sync)) < 0) g_print("Unable to read word clock sync selection: %s\n", snd_strerror(err)); } if (spdif_master.value.integer.value[0]) { if (word_clock_sync.value.integer.value[0]) { toggle_set(hw_master_clock_word_radio, TRUE); } else { toggle_set(hw_master_clock_spdif_radio, TRUE); } } else { toggle_set(hw_master_clock_xtal_radio, TRUE); } master_clock_status_timeout_callback(NULL); } static void master_clock_spdif_master(int on) { int err; spdif_master.value.integer.value[0] = on ? 1 : 0; if ((err = snd_ctl_elem_write(card_ctl, &spdif_master)) < 0) g_print("Unable to write S/PDIF master state: %s\n", snd_strerror(err)); } static void master_clock_word_select(int on) { int err; if (card_eeprom.subvendor != ICE1712_SUBDEVICE_DELTA1010) return; word_clock_sync.value.integer.value[0] = on ? 1 : 0; if ((err = snd_ctl_elem_write(card_ctl, &word_clock_sync)) < 0) g_print("Unable to write word clock sync selection: %s\n", snd_strerror(err)); } void master_clock_toggled(GtkWidget *togglebutton, gpointer data) { char *what = (char *) data; if (!is_active(togglebutton)) return; if (!strcmp(what, "Xtal")) { master_clock_spdif_master(0); } else if (!strcmp(what, "SPDIF")) { master_clock_spdif_master(1); master_clock_word_select(0); } else if (!strcmp(what, "WordClock")) { master_clock_spdif_master(1); master_clock_word_select(1); } else { g_print("master_clock_toggled: %s ???\n", what); } } gint master_clock_status_timeout_callback(gpointer data) { snd_ctl_elem_t sw; int err; if (card_eeprom.subvendor != ICE1712_SUBDEVICE_DELTA1010) return FALSE; memset(&sw, 0, sizeof(sw)); sw.id.iface = SND_CTL_ELEM_IFACE_PCM; strcpy(sw.id.name, "Word Clock Status"); if ((err = snd_ctl_elem_read(card_ctl, &sw)) < 0) g_print("Unable to determine word clock status: %s\n", snd_strerror(err)); gtk_label_set_text(GTK_LABEL(hw_master_clock_status_label), sw.value.integer.value[0] ? "Locked" : "No signal"); return TRUE; } void volume_change_rate_update(void) { int err; if ((err = snd_ctl_elem_read(card_ctl, &volume_rate)) < 0) g_print("Unable to read volume change rate: %s\n", snd_strerror(err)); gtk_adjustment_set_value(GTK_ADJUSTMENT(hw_volume_change_adj), volume_rate.value.integer.value[0]); } void volume_change_rate_adj(GtkAdjustment *adj, gpointer data) { int err; volume_rate.value.integer.value[0] = adj->value; if ((err = snd_ctl_elem_write(card_ctl, &volume_rate)) < 0) g_print("Unable to write volume change rate: %s\n", snd_strerror(err)); } void spdif_output_update(void) { int err, val; if (card_eeprom.subvendor == ICE1712_SUBDEVICE_DELTA44) return; if ((err = snd_ctl_elem_read(card_ctl, &spdif_output)) < 0) g_print("Unable to read Delta S/PDIF output state: %s\n", snd_strerror(err)); val = spdif_output.value.integer.value[0]; if (val & 1) { /* consumer */ toggle_set(hw_spdif_consumer_radio, TRUE); if (val & 8) { toggle_set(hw_consumer_copyright_on_radio, TRUE); } else { toggle_set(hw_consumer_copyright_off_radio, TRUE); } if (val & 0x10) { toggle_set(hw_consumer_emphasis_none_radio, TRUE); } else { toggle_set(hw_consumer_emphasis_5015_radio, TRUE); } switch (val & 0x60) { case 0x00: toggle_set(hw_consumer_category_dat_radio, TRUE); break; case 0x20: toggle_set(hw_consumer_category_pcm_radio, TRUE); break; case 0x40: toggle_set(hw_consumer_category_cd_radio, TRUE); break; case 0x60: toggle_set(hw_consumer_category_general_radio, TRUE); break; } if (val & 0x80) { toggle_set(hw_consumer_copy_1st_radio, TRUE); } else { toggle_set(hw_consumer_copy_original_radio, TRUE); } } else { toggle_set(hw_spdif_professional_radio, TRUE); if (val & 2) { toggle_set(hw_spdif_profi_audio_radio, TRUE); } else { toggle_set(hw_spdif_profi_nonaudio_radio, TRUE); } switch (val & 0x60) { case 0x00: toggle_set(hw_profi_emphasis_ccitt_radio, TRUE); break; case 0x20: toggle_set(hw_profi_emphasis_none_radio, TRUE); break; case 0x40: toggle_set(hw_profi_emphasis_5015_radio, TRUE); break; case 0x60: toggle_set(hw_profi_emphasis_notid_radio, TRUE); break; } if (val & 0x80) { toggle_set(hw_profi_stream_notid_radio, TRUE); } else { toggle_set(hw_profi_stream_stereo_radio, TRUE); } } } static void spdif_output_write(void) { int err; if ((err = snd_ctl_elem_write(card_ctl, &spdif_output)) < 0) g_print("Unable to write Delta S/PDIF Output Defaults: %s\n", snd_strerror(err)); } void profi_data_toggled(GtkWidget *togglebutton, gpointer data) { char *str = (char *)data; int val = spdif_output.value.integer.value[0]; if (!is_active(togglebutton)) return; if (val & 1) return; if (!strcmp(str, "Audio")) { val |= 0x02; } else if (!strcmp(str, "Non-audio")) { val &= ~0x02; } spdif_output.value.integer.value[0] = val; spdif_output_write(); } void profi_stream_toggled(GtkWidget *togglebutton, gpointer data) { char *str = (char *)data; int val = spdif_output.value.integer.value[0]; if (!is_active(togglebutton)) return; if (val & 1) return; if (!strcmp(str, "NOTID")) { val |= 0x80; } else if (!strcmp(str, "Stereo")) { val &= ~0x80; } spdif_output.value.integer.value[0] = val; spdif_output_write(); } void profi_emphasis_toggled(GtkWidget *togglebutton, gpointer data) { char *str = (char *)data; int val = spdif_output.value.integer.value[0]; if (!is_active(togglebutton)) return; if (val & 1) return; if (!strcmp(str, "CCITT")) { val &= ~0x60; } else if (!strcmp(str, "No")) { val &= ~0x60; val |= 0x20; } else if (!strcmp(str, "5015")) { val &= ~0x60; val |= 0x40; } else if (!strcmp(str, "NOTID")) { val |= 0x60; } spdif_output.value.integer.value[0] = val; spdif_output_write(); } void consumer_copyright_toggled(GtkWidget *togglebutton, gpointer data) { char *str = (char *)data; int val = spdif_output.value.integer.value[0]; if (!is_active(togglebutton)) return; if (!(val & 1)) return; if (!strcmp(str, "Copyright")) { val |= 0x08; } else if (!strcmp(str, "Permitted")) { val &= ~0x08; } spdif_output.value.integer.value[0] = val; spdif_output_write(); } void consumer_copy_toggled(GtkWidget *togglebutton, gpointer data) { char *str = (char *)data; int val = spdif_output.value.integer.value[0]; if (!is_active(togglebutton)) return; if (!(val & 1)) return; if (!strcmp(str, "1st")) { val |= 0x80; } else if (!strcmp(str, "Original")) { val &= ~0x80; } spdif_output.value.integer.value[0] = val; spdif_output_write(); } void consumer_emphasis_toggled(GtkWidget *togglebutton, gpointer data) { char *str = (char *)data; int val = spdif_output.value.integer.value[0]; if (!is_active(togglebutton)) return; if (!(val & 1)) return; if (!strcmp(str, "No")) { val |= 0x10; } else if (!strcmp(str, "5015")) { val &= ~0x10; } spdif_output.value.integer.value[0] = val; spdif_output_write(); } void consumer_category_toggled(GtkWidget *togglebutton, gpointer data) { char *str = (char *)data; int val = spdif_output.value.integer.value[0]; if (!is_active(togglebutton)) return; if (!(val & 1)) return; if (!strcmp(str, "DAT")) { val &= ~0x60; } else if (!strcmp(str, "PCM")) { val &= ~0x60; val |= 0x20; } else if (!strcmp(str, "CD")) { val &= ~0x60; val |= 0x40; } else if (!strcmp(str, "General")) { val |= 0x60; } spdif_output.value.integer.value[0] = val; spdif_output_write(); } void spdif_output_toggled(GtkWidget *togglebutton, gpointer data) { char *str = (char *)data; int page; if (is_active(togglebutton)) { if (!strcmp(str, "Professional")) { if (spdif_output.value.integer.value[0] & 0x01) { /* default setup: audio, no emphasis */ spdif_output.value.integer.value[0] = 0x22; } page = 0; } else { if (!(spdif_output.value.integer.value[0] & 0x01)) { /* default setup: no emphasis, PCM encoder */ spdif_output.value.integer.value[0] = 0x31; } page = 1; } spdif_output_write(); gtk_notebook_set_page(GTK_NOTEBOOK(hw_spdif_output_notebook), page); spdif_output_update(); } } void spdif_input_update(void) { int err; if (card_eeprom.subvendor != ICE1712_SUBDEVICE_DELTADIO2496) return; if ((err = snd_ctl_elem_read(card_ctl, &spdif_input)) < 0) g_print("Unable to read S/PDIF input switch: %s\n", snd_strerror(err)); if (spdif_input.value.integer.value[0]) { toggle_set(hw_spdif_input_optical_radio, TRUE); } else { toggle_set(hw_spdif_input_coaxial_radio, TRUE); } } void spdif_input_toggled(GtkWidget *togglebutton, gpointer data) { int err; char *str = (char *)data; if (!is_active(togglebutton)) return; if (!strcmp(str, "Optical")) spdif_input.value.integer.value[0] = 1; else spdif_input.value.integer.value[0] = 0; if ((err = snd_ctl_elem_write(card_ctl, &spdif_input)) < 0) g_print("Unable to write S/PDIF input switch: %s\n", snd_strerror(err)); } void hardware_init(void) { memset(&spdif_master, 0, sizeof(spdif_master)); spdif_master.id.iface = SND_CTL_ELEM_IFACE_PCM; strcpy(spdif_master.id.name, "Multi Track S/PDIF Master"); memset(&word_clock_sync, 0, sizeof(spdif_master)); word_clock_sync.id.iface = SND_CTL_ELEM_IFACE_PCM; strcpy(word_clock_sync.id.name, "Word Clock Sync"); memset(&volume_rate, 0, sizeof(volume_rate)); spdif_master.id.iface = SND_CTL_ELEM_IFACE_PCM; strcpy(volume_rate.id.name, "Multi Track Volume Rate"); memset(&spdif_input, 0, sizeof(spdif_input)); spdif_master.id.iface = SND_CTL_ELEM_IFACE_PCM; strcpy(spdif_input.id.name, "S/PDIF Input Optical"); memset(&spdif_output, 0, sizeof(spdif_output)); spdif_master.id.iface = SND_CTL_ELEM_IFACE_PCM; strcpy(spdif_output.id.name, "Delta S/PDIF Output Defaults"); } void hardware_postinit(void) { master_clock_update(); volume_change_rate_update(); spdif_input_update(); spdif_output_update(); }