亚洲欧美第一页_禁久久精品乱码_粉嫩av一区二区三区免费野_久草精品视频

? 歡迎來到蟲蟲下載站! | ?? 資源下載 ?? 資源專輯 ?? 關(guān)于我們
? 蟲蟲下載站

?? sb_mixer.c

?? LINUX1.0源代碼,代碼條理清晰
?? C
字號:

/*
 * sound/sb_mixer.c
 * 
 * The low level mixer driver for the SoundBlaster Pro and SB16 cards.
 * 
 * Copyright by Hannu Savolainen 1993
 * 
 * Redistribution and use in source and binary forms, with or without
 * modification, are permitted provided that the following conditions are
 * met: 1. Redistributions of source code must retain the above copyright
 * notice, this list of conditions and the following disclaimer. 2.
 * Redistributions in binary form must reproduce the above copyright notice,
 * this list of conditions and the following disclaimer in the documentation
 * and/or other materials provided with the distribution.
 * 
 * THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND ANY
 * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
 * WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
 * DISCLAIMED.  IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE FOR
 * ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
 * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR
 * SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER
 * CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
 * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
 * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
 * SUCH DAMAGE.
 * 
 */

#include "sound_config.h"

#if defined(CONFIGURE_SOUNDCARD) && !defined(EXCLUDE_SB) && !defined(EXCLUDE_SBPRO)
#define __SB_MIXER_C__

#include "sb.h"
#include "sb_mixer.h"
#undef SB_TEST_IRQ

extern int      sbc_base;

static int mixer_initialized = 0;

static int supported_rec_devices;
static int supported_devices;
static int recmask = 0;
static int mixer_model;
static int mixer_caps;
static mixer_tab *iomap;

void
sb_setmixer (unsigned int port, unsigned int value)
{
  unsigned long flags;

  DISABLE_INTR(flags);
  OUTB ((unsigned char)(port & 0xff), MIXER_ADDR);	/* Select register */
  tenmicrosec ();
  OUTB ((unsigned char)(value & 0xff), MIXER_DATA);
  tenmicrosec ();
  RESTORE_INTR(flags);
}

int
sb_getmixer (unsigned int port)
{
  int             val;
  unsigned long flags;

  DISABLE_INTR(flags);
  OUTB ((unsigned char)(port & 0xff), MIXER_ADDR);	/* Select register */
  tenmicrosec ();
  val = INB (MIXER_DATA);
  tenmicrosec ();
  RESTORE_INTR(flags);

  return val;
}

void
sb_mixer_set_stereo(int mode)
{
  if (!mixer_initialized) return;

  sb_setmixer (OUT_FILTER, ((sb_getmixer (OUT_FILTER) & ~STEREO_DAC)
			 | (mode ? STEREO_DAC : MONO_DAC)));
}

static int
detect_mixer (void)
{
  /*
   * Detect the mixer by changing parameters of two volume channels. If the
   * values read back match with the values written, the mixer is there (is
   * it?)
   */
  sb_setmixer (FM_VOL, 0xff);
  sb_setmixer (VOC_VOL, 0x33);

  if (sb_getmixer (FM_VOL) != 0xff)
    return 0;			/* No match */
  if (sb_getmixer (VOC_VOL) != 0x33)
    return 0;

  return 1;
}

static void
change_bits(unsigned char *regval, int dev, int chn, int newval)
{
	unsigned char mask;
	int shift;

	mask = (1 << (*iomap)[dev][chn].nbits)-1;
	newval = ((newval * mask) + 50) / 100;	/* Scale it */

	shift = (*iomap)[dev][chn].bitoffs-(*iomap)[dev][LEFT_CHN].nbits+1;

	*regval &= ~(mask << shift);	/* Filter out the previous value */
	*regval |= (newval & mask) << shift; /* Set the new value */
}

static int
sb_mixer_get(int dev)
{
	if (!((1<<dev) & supported_devices)) 
	   return RET_ERROR(EINVAL);

	return levels[dev];
}

static int
sb_mixer_set (int dev, int value)
{
	int left = value & 0x000000ff;
	int right = (value & 0x0000ff00) >> 8;

	int regoffs;
	unsigned char val;

	if (left > 100) left = 100;
	if (right > 100) right = 100;

	if (dev > 31) return RET_ERROR(EINVAL);

	if (!(supported_devices & (1 << dev)))	/* Not supported */
	   return RET_ERROR(EINVAL);

	regoffs = (*iomap)[dev][LEFT_CHN].regno;

	if (regoffs == 0)
	   return RET_ERROR(EINVAL);

	val = sb_getmixer(regoffs);
	change_bits(&val, dev, LEFT_CHN, left);

	levels[dev] = left|(left << 8);

	if ((*iomap)[dev][RIGHT_CHN].regno != regoffs) /* Change register */
	{
		sb_setmixer(regoffs, val);	/* Save the old one */
		regoffs = (*iomap)[dev][RIGHT_CHN].regno;

		if (regoffs == 0)
		   return left|(left << 8); /* Just left channel present */

		val = sb_getmixer(regoffs);	/* Read the new one */
	}

	change_bits(&val, dev, RIGHT_CHN, right);
	sb_setmixer(regoffs, val);

	levels[dev] = left | (right << 8);
	return left | (right << 8);
}

static void
set_recsrc(int src)
{
	sb_setmixer(RECORD_SRC, (sb_getmixer(RECORD_SRC)&~7) | (src&0x7));
}

static int
set_recmask(int mask)
{
      int devmask, i;
      unsigned char regimageL, regimageR;

      devmask = mask & supported_rec_devices;

      switch (mixer_model)
      {
      case 3:

      if (devmask != SOUND_MASK_MIC &&
	  devmask != SOUND_MASK_LINE &&
	  devmask != SOUND_MASK_CD)
	{			/* More than one devices selected. Drop the
				 * previous selection */
	  devmask &= ~recmask;
	}

      if (devmask != SOUND_MASK_MIC &&
	  devmask != SOUND_MASK_LINE &&
	  devmask != SOUND_MASK_CD)
	{			/* More than one devices selected. Default to
				 * mic */
	  devmask = SOUND_MASK_MIC;
	}


      if (devmask ^ recmask)/* Input source changed */
	{
	  switch (devmask)
	    {

	    case SOUND_MASK_MIC:
	      set_recsrc (SRC_MIC);
	      break;

	    case SOUND_MASK_LINE:
	      set_recsrc (SRC_LINE);
	      break;

	    case SOUND_MASK_CD:
	      set_recsrc (SRC_CD);
	      break;

	    default:
	      set_recsrc (SRC_MIC);
	    }
	}

      break;

      case 4:
	if (!devmask) devmask = SOUND_MASK_MIC;

        regimageL = regimageR = 0;
	for (i=0;i<SOUND_MIXER_NRDEVICES;i++)
	  if ((1<<i) & devmask)
	  {
	    regimageL |= sb16_recmasks_L[i];
	    regimageR |= sb16_recmasks_R[i];
	  }
	sb_setmixer(SB16_IMASK_L, regimageL);
	sb_setmixer(SB16_IMASK_R, regimageR);
      break;
      }

      recmask = devmask;
      return recmask;
}

static int
sb_mixer_ioctl (int dev, unsigned int cmd, unsigned int arg)
{
  if (((cmd >> 8) & 0xff) == 'M')
    {
      if (cmd & IOC_IN)
        switch (cmd & 0xff)
        {
        case SOUND_MIXER_RECSRC:
           return IOCTL_OUT(arg, set_recmask(IOCTL_IN(arg)));
           break;

	default:
	   return IOCTL_OUT (arg, sb_mixer_set (cmd & 0xff, IOCTL_IN (arg)));
        }
      else
	  switch (cmd & 0xff)	/* Return parameters */
	    {

	    case SOUND_MIXER_RECSRC:
	      return IOCTL_OUT (arg, recmask);
	      break;

	    case SOUND_MIXER_DEVMASK:
	      return IOCTL_OUT (arg, supported_devices);
	      break;

	    case SOUND_MIXER_STEREODEVS:
	      return IOCTL_OUT (arg, supported_devices & 
	                              ~(SOUND_MASK_MIC|SOUND_MASK_SPEAKER));
	      break;

	    case SOUND_MIXER_RECMASK:
	      return IOCTL_OUT (arg, supported_rec_devices);
	      break;

	    case SOUND_MIXER_CAPS:
	      return IOCTL_OUT (arg, mixer_caps);
	      break;

	    default:
	      return IOCTL_OUT (arg, sb_mixer_get (cmd & 0xff));
	    }
    }
  else
    return RET_ERROR (EINVAL);
}

static struct mixer_operations sb_mixer_operations =
{
  sb_mixer_ioctl
};

static void
sb_mixer_reset(void)
{
  int i;

  for (i = 0; i < SOUND_MIXER_NRDEVICES; i++)
    sb_mixer_set (i, levels[i]);
  set_recmask(SOUND_MASK_MIC);
}

void
sb_mixer_init(int major_model)
{
	sb_setmixer(0x00, 0);	/* Reset mixer */

	if (!detect_mixer()) return;	/* No mixer. Why? */

	mixer_initialized = 1;
	mixer_model = major_model;

        switch (major_model)
        {
        case 3:
	  mixer_caps = SOUND_CAP_EXCL_INPUT;
	  supported_devices = SBPRO_MIXER_DEVICES;
	  supported_rec_devices = SBPRO_RECORDING_DEVICES;
	  iomap = &sbpro_mix;
	  break;

	case 4:
	  mixer_caps = 0;
	  supported_devices = SB16_MIXER_DEVICES;
	  supported_rec_devices = SB16_RECORDING_DEVICES;
	  iomap = &sb16_mix;
	  break;

	default:
	  printk("SB Warning: Unsupported mixer type\n");
	  return;
	}

        mixer_devs[num_mixers++] = &sb_mixer_operations;
        sb_mixer_reset();
}

#endif

?? 快捷鍵說明

復(fù)制代碼 Ctrl + C
搜索代碼 Ctrl + F
全屏模式 F11
切換主題 Ctrl + Shift + D
顯示快捷鍵 ?
增大字號 Ctrl + =
減小字號 Ctrl + -
亚洲欧美第一页_禁久久精品乱码_粉嫩av一区二区三区免费野_久草精品视频
91最新地址在线播放| 亚洲国产精品一区二区久久| 久久99精品国产| 91精品国产手机| 麻豆视频一区二区| 欧美电视剧免费观看| 国产一区在线不卡| 国产日韩在线不卡| av午夜精品一区二区三区| 亚洲少妇屁股交4| 欧美色手机在线观看| 亚洲成av人在线观看| 欧美一区二区三区在线视频| 日韩福利视频导航| 久久影院午夜片一区| 成人av在线资源网| 亚洲成国产人片在线观看| 88在线观看91蜜桃国自产| 韩国成人精品a∨在线观看| 中文字幕精品一区| 欧洲av在线精品| 麻豆国产精品777777在线| 国产色爱av资源综合区| 99精品久久只有精品| 天堂久久一区二区三区| 2017欧美狠狠色| 95精品视频在线| 日韩主播视频在线| 国产欧美一区二区精品性色| 欧美最猛黑人xxxxx猛交| 久久狠狠亚洲综合| 亚洲欧美激情小说另类| 日韩亚洲欧美一区| 91在线观看成人| 秋霞av亚洲一区二区三| 亚洲欧美综合另类在线卡通| 日韩一区二区在线看| 99久免费精品视频在线观看| 日韩激情视频网站| 亚洲欧洲韩国日本视频| 精品久久一二三区| 色噜噜狠狠一区二区三区果冻| 男女男精品视频| 一个色综合av| 国产欧美1区2区3区| 69久久夜色精品国产69蝌蚪网| 成人av中文字幕| 精品无人码麻豆乱码1区2区| 亚洲成在人线在线播放| 亚洲欧美在线观看| 久久久91精品国产一区二区精品| 欧洲日韩一区二区三区| 成人av网站在线观看| 久久99日本精品| 性欧美疯狂xxxxbbbb| 日韩伦理av电影| 国产视频一区在线播放| 日韩欧美亚洲国产精品字幕久久久| 91在线精品一区二区三区| 国产美女精品一区二区三区| 日本少妇一区二区| 亚洲一区在线观看网站| 综合激情网...| 国产精品女同互慰在线看| 精品国产一二三区| 7777精品伊人久久久大香线蕉经典版下载| www.日韩在线| 国产成人丝袜美腿| 国产精品1024久久| 国产一区二区三区精品欧美日韩一区二区三区 | 波波电影院一区二区三区| 国产在线精品一区二区夜色 | 日韩一二三区视频| 欧美日韩成人综合| 精品1区2区3区| 欧美午夜电影网| 欧美性受极品xxxx喷水| 色婷婷久久一区二区三区麻豆| av一区二区三区黑人| 成人美女在线观看| 波多野结衣中文一区| 波多野结衣视频一区| www.欧美日韩国产在线| 色综合久久久久综合99| 日本道免费精品一区二区三区| 91网站在线播放| 色成人在线视频| 欧美色大人视频| 日韩欧美一区在线| 精品久久久久久综合日本欧美| 精品久久人人做人人爰| 国产香蕉久久精品综合网| 国产亚洲综合性久久久影院| 久久精品亚洲精品国产欧美| 日本一区二区免费在线观看视频| 亚洲国产成人午夜在线一区| 欧美国产精品久久| 亚洲人成影院在线观看| 亚洲综合图片区| 日韩电影在线免费| 国产一区二区在线观看免费 | 91小视频免费看| 欧美怡红院视频| 欧美日韩成人一区| 欧美电影精品一区二区| 国产精品色在线| 一区二区三区四区不卡视频| 午夜成人免费电影| 紧缚捆绑精品一区二区| 99riav久久精品riav| 欧美精品久久一区| 日本一区二区三区在线观看| 亚洲男人天堂一区| 美女在线视频一区| proumb性欧美在线观看| 88在线观看91蜜桃国自产| 国产欧美日韩亚州综合| 夜夜嗨av一区二区三区| 免费成人av资源网| 99这里都是精品| 欧美精品一卡二卡| 国产精品久久久久久福利一牛影视| 亚洲一区二区美女| 国产精品一区二区三区四区| 在线一区二区三区| 精品捆绑美女sm三区| 亚洲综合一区在线| 高清不卡在线观看av| 日韩限制级电影在线观看| 亚洲色图视频网站| 国产乱码字幕精品高清av| 欧美色精品在线视频| 国产精品理论片| 激情六月婷婷久久| 欧美日本不卡视频| 亚洲裸体在线观看| 国产精品亚洲成人| 日韩精品一区二区三区老鸭窝| 亚洲欧美日韩中文播放| 国产成人免费视频网站| 日韩视频免费观看高清完整版 | 欧美成人一区二区三区在线观看| 国产精品久久影院| 久久国产精品露脸对白| 色94色欧美sute亚洲线路一久| 欧美xxx久久| 香港成人在线视频| 91亚洲精品久久久蜜桃网站 | 99精品热视频| 国产欧美一区视频| 奇米一区二区三区av| 色狠狠av一区二区三区| 日韩一级完整毛片| 日韩成人午夜精品| 日本韩国视频一区二区| 国产日韩精品一区二区三区 | 国产乱码精品一区二区三区五月婷| 欧美一级午夜免费电影| 日本福利一区二区| 精品国产一区二区三区四区四| 国产精品久久久久永久免费观看| 亚洲国产乱码最新视频| 狠狠色狠狠色综合系列| 欧美一区三区二区| 亚洲人成精品久久久久久| 狠狠色丁香久久婷婷综| av亚洲精华国产精华精| 精品久久久久久久人人人人传媒 | 色综合久久综合中文综合网| 欧美一区二区三区在线视频| 日韩精品一二区| 色综合色综合色综合色综合色综合| 久久丝袜美腿综合| 久久99国产精品麻豆| 欧美精品少妇一区二区三区| 亚洲欧洲制服丝袜| 国产精品99久| 中文字幕久久午夜不卡| 狠狠色丁香婷婷综合| 欧美一个色资源| 亚洲图片你懂的| 成人免费不卡视频| 国产色一区二区| 日本欧美加勒比视频| 欧美日韩一级片网站| 亚洲韩国一区二区三区| 成人福利电影精品一区二区在线观看| 国产日产亚洲精品系列| 国产一区91精品张津瑜| 久久这里只精品最新地址| 精品伊人久久久久7777人| 国产三级一区二区三区| 国产精品77777竹菊影视小说| 精品国产3级a| 韩国中文字幕2020精品| 久久久久99精品一区| 国产成人精品免费看| 国产精品久久久久久久岛一牛影视 | 婷婷成人综合网| 7777精品伊人久久久大香线蕉超级流畅 |