Logo Search packages:      
Sourcecode: alsa-tools version File versions  Download package

HDSPMixerCard.cxx

/*
 *   HDSPMixer
 *    
 *   Copyright (C) 2003 Thomas Charbonnel (thomas@undata.org)
 *    
 *   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.
 */

#pragma implementation
#include "HDSPMixerCard.h"

static char channel_map_df_ss[26] = {
    0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15, 16, 17,
    18, 19, 20, 21, 22, 23, 24, 25
};

static char channel_map_mf_ss[26] = {
    0, 1, 2, 3, 4, 5, 6, 7,
    16, 17, 18, 19, 20, 21, 22, 23, 
    24, 25,
    -1, -1, -1, -1, -1, -1, -1, -1
};

static char meter_map_ds[26] = {
    0, 1, 2, 3, 8, 9, 10, 11, 16, 17, 18, 19, 
    24, 25,
    -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1
};

static char channel_map_ds[26] = {
    1, 3, 5, 7, 9, 11, 13, 15, 17, 19, 21, 23, 
    24, 25,
    -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1
};

static char dest_map_mf_ss[10] = {
    0, 2, 4, 6, 16, 18, 20, 22, 24, 26 
};

static char dest_map_ds[8] = {
    0, 2, 8, 10, 16, 18, 24, 26 
};

static char dest_map_df_ss[14] = {
    0, 2, 4, 6, 8, 10, 12, 14, 16, 18, 20, 22, 24, 26 
};

static char dest_map_h9652_ss[13] = {
    0, 2, 4, 6, 8, 10, 12, 14, 16, 18, 20, 22, 24 
};

static char dest_map_h9652_ds[7] = {
    0, 2, 8, 10, 16, 18, 24 
};

static char dest_map_h9632_ss[8] = {
    0, 2, 4, 6, 8, 10, 12, 14
};

static char dest_map_h9632_ds[6] = {
    0, 2, 8, 10, 12, 14
};

static char dest_map_h9632_qs[4] = {
    8, 10, 12, 14
};

static char channel_map_h9632_ss[16] = {
    0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15
};

static char channel_map_h9632_ds[12] = {
    0, 1, 2, 3, 8, 9, 10, 11, 12, 13, 14, 15
};

static char channel_map_h9632_qs[8] = {
    8, 9, 10, 11, 12, 13, 14, 15
};

static void alsactl_cb(snd_async_handler_t *handler)
{
    int err, clock_value;
    snd_ctl_t *ctl;
    snd_ctl_event_t *event;
    snd_ctl_elem_value_t *elemval;
    snd_ctl_elem_id_t *elemid;
    HDSPMixerCard *card;
    
    card = (HDSPMixerCard *)snd_async_handler_get_callback_private(handler);
    
    snd_ctl_elem_value_alloca(&elemval);
    snd_ctl_elem_id_alloca(&elemid);
    
    ctl = snd_async_handler_get_ctl(handler);
    
    if ((err = snd_ctl_nonblock(ctl, 1))) {
      printf("Error setting non blocking mode for card %s\n", card->name);
      return;
    }
    
    snd_ctl_event_malloc(&event);    
    
    while ((err = snd_ctl_read(ctl, event)) > 0) {
      if (snd_ctl_event_elem_get_numid(event) == 11 && (card == card->basew->cards[card->basew->current_card])) {
          /* We have a clock change and are the focused card */
          snd_ctl_event_elem_get_id(event, elemid);
          snd_ctl_elem_value_set_id(elemval, elemid);
          if ((err = snd_ctl_elem_read(ctl, elemval)) < 0) {
            fprintf(stderr, "Error reading snd_ctl_elem_t\n");
            snd_ctl_event_free(event);
            return;
          }
          clock_value = snd_ctl_elem_value_get_enumerated(elemval, 0);
          if (clock_value == 0) {
            int new_speed = card->getAutosyncSpeed();
            if (new_speed >= 0 && new_speed != card->speed_mode) card->setMode(new_speed);
          }
          if (clock_value > 3 && clock_value < 7 && card->speed_mode != 1) {
            card->setMode(1);
          } else if (clock_value < 4 && card->speed_mode != 0) {
            card->setMode(0);
          } else if (clock_value > 6 && card->speed_mode != 2) {
            card->setMode(2);
          }
      }
      snd_ctl_event_clear(event);
    }
    
    snd_ctl_event_free(event);
}

int HDSPMixerCard::getAutosyncSpeed()
{
    int err, rate;
    snd_ctl_elem_value_t *elemval;
    snd_ctl_elem_id_t * elemid;
    snd_ctl_t *handle;
    snd_ctl_elem_value_alloca(&elemval);
    snd_ctl_elem_id_alloca(&elemid);
    if ((err = snd_ctl_open(&handle, name, SND_CTL_NONBLOCK)) < 0) {
      fprintf(stderr, "Error accessing ctl interface on card %s\n.", name);
      return -1; 
    }
    
    snd_ctl_elem_id_set_name(elemid, "System Sample Rate");
    snd_ctl_elem_id_set_interface(elemid, SND_CTL_ELEM_IFACE_MIXER);
    snd_ctl_elem_id_set_index(elemid, 0);
    snd_ctl_elem_value_set_id(elemval, elemid);
    if (snd_ctl_elem_read(handle, elemval) < 0) {
      snd_ctl_elem_id_set_interface(elemid, SND_CTL_ELEM_IFACE_HWDEP);
      snd_ctl_elem_value_set_id(elemval, elemid);
      snd_ctl_elem_read(handle, elemval);
    }
    rate = snd_ctl_elem_value_get_integer(elemval, 0);

    snd_ctl_close(handle);

    if (rate > 96000) {
      return 2;
    } else if (rate > 48000) {
      return 1;
    }
    return 0;
}

int HDSPMixerCard::getSpeed()
{
    int err, val;
    snd_ctl_elem_value_t *elemval;
    snd_ctl_elem_id_t * elemid;
    snd_ctl_t *handle;
    snd_ctl_elem_value_alloca(&elemval);
    snd_ctl_elem_id_alloca(&elemid);
    if ((err = snd_ctl_open(&handle, name, SND_CTL_NONBLOCK)) < 0) {
      fprintf(stderr, "Error accessing ctl interface on card %s\n.", name);
      return -1; 
    }
    snd_ctl_elem_id_set_name(elemid, "Sample Clock Source");
    snd_ctl_elem_id_set_interface(elemid, SND_CTL_ELEM_IFACE_MIXER);
    snd_ctl_elem_id_set_index(elemid, 0);
    snd_ctl_elem_value_set_id(elemval, elemid);
    if (snd_ctl_elem_read(handle, elemval) < 0) {
      snd_ctl_elem_id_set_interface(elemid, SND_CTL_ELEM_IFACE_PCM);
      snd_ctl_elem_value_set_id(elemval, elemid);
      snd_ctl_elem_read(handle, elemval);
    }
    val = snd_ctl_elem_value_get_enumerated(elemval, 0);
    snd_ctl_close(handle);
    switch (val) {
    case 0:
      /* Autosync mode : We need to determine sample rate */
      return getAutosyncSpeed();
      break;
    case 1:
    case 2:
    case 3:
      /* SR <= 48000 - normal speed */
      return 0;
    case 4:
    case 5:
    case 6:
      /* SR > 48000 Hz - double speed */
      return 1;
    case 7:
    case 8:
    case 9:
      /* SR > 96000 Hz - quad speed */
      return 2;    
    default:
      /* Should never happen */
      return 0;
    }
    return 0;    
}

HDSPMixerCard::HDSPMixerCard(HDSP_IO_Type cardtype, int id)
{
    type = cardtype;
    card_id = id;
    snprintf(name, 6, "hw:%i", card_id);
    h9632_aeb.aebi = 0;
    h9632_aeb.aebo = 0;
    if (type == H9632) {
      getAeb();
      playbacks_offset = 16;
    } else {
      playbacks_offset = 26;
    }
    speed_mode = getSpeed();
    if (speed_mode < 0) {
      fprintf(stderr, "Error trying to determine speed mode for card %s, exiting.\n", name);
      exit(EXIT_FAILURE);
    }
    
    /* Set channels and mappings */
    adjustSettings();
        
    basew = NULL;
}

void HDSPMixerCard::getAeb() {
    int err;
    snd_hwdep_t *hw;
    snd_hwdep_info_t *info;
    snd_hwdep_info_alloca(&info);
    if ((err = snd_hwdep_open(&hw, name, SND_HWDEP_OPEN_DUPLEX)) != 0) {
      fprintf(stderr, "Error opening hwdep device on card %s.\n", name);
      return; 
    }
    if ((err = snd_hwdep_ioctl(hw, SNDRV_HDSP_IOCTL_GET_9632_AEB, &h9632_aeb)) < 0) {
      fprintf(stderr, "Hwdep ioctl error on card %s : %s.\n", name, snd_strerror(err));
      snd_hwdep_close(hw);
      return; 
    }
    snd_hwdep_close(hw);
}

void HDSPMixerCard::adjustSettings() {
    if (type == Multiface) {
      switch (speed_mode) {
      case 0:
          channels = 18;
          channel_map = channel_map_mf_ss;
          dest_map = dest_map_mf_ss;
          meter_map = channel_map_mf_ss;
          lineouts = 2;
          break;
      case 1:
          channels = 14;
          channel_map = meter_map_ds;
          dest_map = dest_map_ds;
          meter_map = meter_map_ds;
          lineouts = 2;
          break;
      case 2:
          /* should never happen */
          break;
      }
    } else if (type == Digiface) {
      switch (speed_mode) {
      case 0:
          channels = 26;
          channel_map = channel_map_df_ss;
          dest_map = dest_map_df_ss;
          meter_map = channel_map_df_ss;
          lineouts = 2;
          break;
      case 1:
          channels = 14;
          channel_map = channel_map_ds;
          dest_map = dest_map_ds;
          meter_map = meter_map_ds;
          lineouts = 2;
          break;
      case 2:
          /* should never happen */
          break;
      }
    } else if (type == H9652) {
      switch (speed_mode) {
      case 0:
          channels = 26;
          channel_map = channel_map_df_ss;
          dest_map = dest_map_h9652_ss;
          meter_map = channel_map_df_ss;
          lineouts = 0;
          break;
      case 1:
          channels = 14;
          channel_map = channel_map_ds;
          dest_map = dest_map_h9652_ds;
          meter_map = meter_map_ds;
          lineouts = 0;
          break;
      case 2:
          /* should never happen */
          break;
      }
    } else if (type == H9632) {
      switch (speed_mode) {
      case 0:
          channels = 12 + ((h9632_aeb.aebi || h9632_aeb.aebo) ? 4 : 0);
          channel_map = channel_map_h9632_ss;
          dest_map = dest_map_h9632_ss;
          meter_map = channel_map_h9632_ss;
          lineouts = 0;
          break;
      case 1:
          channels = 8 + ((h9632_aeb.aebi || h9632_aeb.aebo) ? 4 : 0);
          channel_map = channel_map_h9632_ds;
          dest_map = dest_map_h9632_ds;
          meter_map = channel_map_h9632_ds;
          lineouts = 0;
          break;
      case 2:
          channels = 4 + ((h9632_aeb.aebi || h9632_aeb.aebo) ? 4 : 0);
          channel_map = channel_map_h9632_qs;
          dest_map = dest_map_h9632_qs;
          meter_map = channel_map_h9632_qs;
          lineouts = 0;
          break;
      }
    }
    window_width = (channels+2)*STRIP_WIDTH;
    window_height = FULLSTRIP_HEIGHT*2+SMALLSTRIP_HEIGHT+MENU_HEIGHT;
} 

void HDSPMixerCard::setMode(int mode)
{
    speed_mode = mode;
    adjustSettings();
    actualizeStrips();

    for (int i = 0; i < channels; ++i) {
      basew->inputs->strips[i]->targets->setLabels();
      basew->playbacks->strips[i]->targets->setLabels();
      basew->outputs->strips[i]->setLabels();
    }
    for (int i = channels; i < channels+lineouts; ++i) {
      basew->outputs->strips[i]->setLabels();    
    }
    if (h9632_aeb.aebo && !h9632_aeb.aebi) {
      basew->inputs->empty_aebi[0]->position(STRIP_WIDTH*(channels-4), basew->inputs->empty_aebi[0]->y());
      basew->inputs->empty_aebi[1]->position(STRIP_WIDTH*(channels-2), basew->inputs->empty_aebi[1]->y());
    } else if (h9632_aeb.aebi && !h9632_aeb.aebo) {
      basew->playbacks->empty_aebo[0]->position(STRIP_WIDTH*(channels-4), basew->playbacks->empty_aebo[0]->y());
      basew->playbacks->empty_aebo[1]->position(STRIP_WIDTH*(channels-2), basew->playbacks->empty_aebo[1]->y());
      basew->outputs->empty_aebo[0]->position(STRIP_WIDTH*(channels-4), basew->outputs->empty_aebo[0]->y());
      basew->outputs->empty_aebo[1]->position(STRIP_WIDTH*(channels-2), basew->outputs->empty_aebo[1]->y());
    }
    basew->inputs->buttons->position(STRIP_WIDTH*channels, basew->inputs->buttons->y());
    basew->inputs->init_sizes();
    basew->playbacks->empty->position(STRIP_WIDTH*channels, basew->playbacks->empty->y());
    basew->playbacks->init_sizes();
    basew->outputs->empty->position(STRIP_WIDTH*(channels+lineouts), basew->outputs->empty->y());    
    basew->outputs->init_sizes();
    basew->inputs->size(window_width, basew->inputs->h());
    basew->playbacks->size(window_width, basew->playbacks->h());
    basew->outputs->size(window_width, basew->outputs->h());
    basew->scroll->init_sizes();
    ((Fl_Widget *)(basew->menubar))->size(window_width, basew->menubar->h());
    basew->size_range(MIN_WIDTH, MIN_HEIGHT, window_width, window_height);
    basew->resize(basew->x(), basew->y(), window_width, basew->h());
    basew->reorder();
    basew->resetMixer();
    basew->inputs->buttons->presets->preset_change(1);
}

void HDSPMixerCard::actualizeStrips()
{
    for (int i = 0; i < HDSP_MAX_CHANNELS; ++i) {
      if (i < channels) {
          basew->inputs->strips[i]->show();
          basew->playbacks->strips[i]->show();
          basew->outputs->strips[i]->show();
      } else {
          basew->inputs->strips[i]->hide();
          basew->playbacks->strips[i]->hide();
          basew->outputs->strips[i]->hide();
      }
    }
    for (int i = channels; i < channels+2; ++i) {
      if (i < channels+lineouts) {
          basew->outputs->strips[i]->show();
      } else {
          basew->outputs->strips[i]->hide();
      }
    }
    if (h9632_aeb.aebi && !h9632_aeb.aebo) {
      for (int i = 0; i < 2; ++i) {
          basew->inputs->empty_aebi[i]->hide();
          basew->playbacks->empty_aebo[i]->show();
          basew->outputs->empty_aebo[i]->show();
      }
      for (int i = channels-4; i < channels; ++i) {
          basew->playbacks->strips[i]->hide();
          basew->outputs->strips[i]->hide();
      }
    } else if (h9632_aeb.aebo && !h9632_aeb.aebi) { 
      for (int i = 0; i < 2; ++i) {
          basew->inputs->empty_aebi[i]->show();
          basew->playbacks->empty_aebo[i]->hide();
          basew->outputs->empty_aebo[i]->hide();
      }        
      for (int i = channels-4; i < channels; ++i) {
          basew->inputs->strips[i]->hide();
      }
    } else {
      for (int i = 0; i < 2; ++i) {
          basew->inputs->empty_aebi[i]->hide();
          basew->playbacks->empty_aebo[i]->hide();
          basew->outputs->empty_aebo[i]->hide();
      }
    }
    if (type != H9652 && type != H9632) basew->outputs->empty->hide();
}

int HDSPMixerCard::initializeCard(HDSPMixerWindow *w)
{
    int err;
    if ((err = snd_ctl_open(&cb_handle, name, SND_CTL_NONBLOCK)) < 0) {
      fprintf(stderr, "Error opening ctl interface for card %s - exiting\n", name);
      exit(EXIT_FAILURE);
    }
    if ((err = snd_async_add_ctl_handler(&cb_handler, cb_handle, alsactl_cb, this)) < 0) {
      fprintf(stderr, "Error registering async ctl callback for card %s - exiting\n", name);
      exit(EXIT_FAILURE);
    }
    if ((err = snd_ctl_subscribe_events(cb_handle, 1)) < 0) {
      fprintf(stderr, "Error subscribing to ctl events for card %s - exiting\n", name);
      exit(EXIT_FAILURE);
    }
    basew = w;
    actualizeStrips();
    return 0;
}


Generated by  Doxygen 1.6.0   Back to index