📄 gus_io.c
字号:
/* * Copyright (c) by Jaroslav Kysela <perex@suse.cz> * I/O routines for GF1/InterWave synthesizer chips * * * 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 <sound/driver.h>#include <linux/delay.h>#include <linux/time.h>#include <sound/core.h>#include <sound/gus.h>void snd_gf1_delay(snd_gus_card_t * gus){ int i; for (i = 0; i < 6; i++) { mb(); inb(GUSP(gus, DRAM)); }}/* * ======================================================================= *//* * ok.. stop of control registers (wave & ramp) need some special things.. * big UltraClick (tm) elimination... */static inline void __snd_gf1_ctrl_stop(snd_gus_card_t * gus, unsigned char reg){ unsigned char value; outb(reg | 0x80, gus->gf1.reg_regsel); mb(); value = inb(gus->gf1.reg_data8); mb(); outb(reg, gus->gf1.reg_regsel); mb(); outb((value | 0x03) & ~(0x80 | 0x20), gus->gf1.reg_data8); mb();}static inline void __snd_gf1_write8(snd_gus_card_t * gus, unsigned char reg, unsigned char data){ outb(reg, gus->gf1.reg_regsel); mb(); outb(data, gus->gf1.reg_data8); mb();}static inline unsigned char __snd_gf1_look8(snd_gus_card_t * gus, unsigned char reg){ outb(reg, gus->gf1.reg_regsel); mb(); return inb(gus->gf1.reg_data8);}static inline void __snd_gf1_write16(snd_gus_card_t * gus, unsigned char reg, unsigned int data){ outb(reg, gus->gf1.reg_regsel); mb(); outw((unsigned short) data, gus->gf1.reg_data16); mb();}static inline unsigned short __snd_gf1_look16(snd_gus_card_t * gus, unsigned char reg){ outb(reg, gus->gf1.reg_regsel); mb(); return inw(gus->gf1.reg_data16);}static inline void __snd_gf1_adlib_write(snd_gus_card_t * gus, unsigned char reg, unsigned char data){ outb(reg, gus->gf1.reg_timerctrl); inb(gus->gf1.reg_timerctrl); inb(gus->gf1.reg_timerctrl); outb(data, gus->gf1.reg_timerdata); inb(gus->gf1.reg_timerctrl); inb(gus->gf1.reg_timerctrl);}static inline void __snd_gf1_write_addr(snd_gus_card_t * gus, unsigned char reg, unsigned int addr, int w_16bit){ if (gus->gf1.enh_mode) { if (w_16bit) addr = ((addr >> 1) & ~0x0000000f) | (addr & 0x0000000f); __snd_gf1_write8(gus, SNDRV_GF1_VB_UPPER_ADDRESS, (unsigned char) ((addr >> 26) & 0x03)); } else if (w_16bit) addr = (addr & 0x00c0000f) | ((addr & 0x003ffff0) >> 1); __snd_gf1_write16(gus, reg, (unsigned short) (addr >> 11)); __snd_gf1_write16(gus, reg + 1, (unsigned short) (addr << 5));}static inline unsigned int __snd_gf1_read_addr(snd_gus_card_t * gus, unsigned char reg, short w_16bit){ unsigned int res; res = ((unsigned int) __snd_gf1_look16(gus, reg | 0x80) << 11) & 0xfff800; res |= ((unsigned int) __snd_gf1_look16(gus, (reg + 1) | 0x80) >> 5) & 0x0007ff; if (gus->gf1.enh_mode) { res |= (unsigned int) __snd_gf1_look8(gus, SNDRV_GF1_VB_UPPER_ADDRESS | 0x80) << 26; if (w_16bit) res = ((res << 1) & 0xffffffe0) | (res & 0x0000000f); } else if (w_16bit) res = ((res & 0x001ffff0) << 1) | (res & 0x00c0000f); return res;}/* * ======================================================================= */void snd_gf1_ctrl_stop(snd_gus_card_t * gus, unsigned char reg){ __snd_gf1_ctrl_stop(gus, reg);}void snd_gf1_write8(snd_gus_card_t * gus, unsigned char reg, unsigned char data){ __snd_gf1_write8(gus, reg, data);}unsigned char snd_gf1_look8(snd_gus_card_t * gus, unsigned char reg){ return __snd_gf1_look8(gus, reg);}void snd_gf1_write16(snd_gus_card_t * gus, unsigned char reg, unsigned int data){ __snd_gf1_write16(gus, reg, data);}unsigned short snd_gf1_look16(snd_gus_card_t * gus, unsigned char reg){ return __snd_gf1_look16(gus, reg);}void snd_gf1_adlib_write(snd_gus_card_t * gus, unsigned char reg, unsigned char data){ __snd_gf1_adlib_write(gus, reg, data);}void snd_gf1_write_addr(snd_gus_card_t * gus, unsigned char reg, unsigned int addr, short w_16bit){ __snd_gf1_write_addr(gus, reg, addr, w_16bit);}unsigned int snd_gf1_read_addr(snd_gus_card_t * gus, unsigned char reg, short w_16bit){ return __snd_gf1_read_addr(gus, reg, w_16bit);}/* */void snd_gf1_i_ctrl_stop(snd_gus_card_t * gus, unsigned char reg){ unsigned long flags; spin_lock_irqsave(&gus->reg_lock, flags); __snd_gf1_ctrl_stop(gus, reg); spin_unlock_irqrestore(&gus->reg_lock, flags);}void snd_gf1_i_write8(snd_gus_card_t * gus, unsigned char reg, unsigned char data){ unsigned long flags; spin_lock_irqsave(&gus->reg_lock, flags); __snd_gf1_write8(gus, reg, data); spin_unlock_irqrestore(&gus->reg_lock, flags);}unsigned char snd_gf1_i_look8(snd_gus_card_t * gus, unsigned char reg){ unsigned long flags; unsigned char res; spin_lock_irqsave(&gus->reg_lock, flags); res = __snd_gf1_look8(gus, reg); spin_unlock_irqrestore(&gus->reg_lock, flags); return res;}void snd_gf1_i_write16(snd_gus_card_t * gus, unsigned char reg, unsigned int data){ unsigned long flags; spin_lock_irqsave(&gus->reg_lock, flags); __snd_gf1_write16(gus, reg, data); spin_unlock_irqrestore(&gus->reg_lock, flags);}unsigned short snd_gf1_i_look16(snd_gus_card_t * gus, unsigned char reg){ unsigned long flags; unsigned short res; spin_lock_irqsave(&gus->reg_lock, flags); res = __snd_gf1_look16(gus, reg); spin_unlock_irqrestore(&gus->reg_lock, flags); return res;}#if 0void snd_gf1_i_adlib_write(snd_gus_card_t * gus, unsigned char reg, unsigned char data){ unsigned long flags; spin_lock_irqsave(&gus->reg_lock, flags); __snd_gf1_adlib_write(gus, reg, data); spin_unlock_irqrestore(&gus->reg_lock, flags);}void snd_gf1_i_write_addr(snd_gus_card_t * gus, unsigned char reg, unsigned int addr, short w_16bit){ unsigned long flags; spin_lock_irqsave(&gus->reg_lock, flags); __snd_gf1_write_addr(gus, reg, addr, w_16bit); spin_unlock_irqrestore(&gus->reg_lock, flags);}#endif /* 0 */
⌨️ 快捷键说明
复制代码
Ctrl + C
搜索代码
Ctrl + F
全屏模式
F11
切换主题
Ctrl + Shift + D
显示快捷键
?
增大字号
Ctrl + =
减小字号
Ctrl + -