/*****************************************************************************/ /* * sm.c -- soundcard radio modem driver. * * Copyright (C) 1996-2000 Thomas Sailer (sailer@ife.ee.ethz.ch) * * 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. * * Please note that the GPL allows you to use the driver, NOT the radio. * In order to use the radio, you need a license from the communications * authority of your country. * * * Command line options (insmod command line) * * mode mode string; eg. "wss:afsk1200" * iobase base address of the soundcard; common values are 0x220 for sbc, * 0x530 for wss * irq interrupt number; common values are 7 or 5 for sbc, 11 for wss * dma dma number; common values are 0 or 1 * * * History: * 0.1 21.09.1996 Started * 18.10.1996 Changed to new user space access routines (copy_{to,from}_user) * 0.4 21.01.1997 Separately compileable soundcard/modem modules * 0.5 03.03.1997 fixed LPT probing (check_lpt result was interpreted the wrong way round) * 0.6 16.04.1997 init code/data tagged * 0.7 30.07.1997 fixed halfduplex interrupt handlers/hotfix for CS423X * 0.8 14.04.1998 cleanups * 0.9 03.08.1999 adapt to Linus' new __setup/__initcall * use parport lowlevel drivers instead of directly writing to a parallel port * removed some pre-2.2 kernel compatibility cruft * 0.10 10.08.1999 Check if parport can do SPP and is safe to access during interrupt contexts * 0.11 12.02.2000 adapted to softnet driver interface * 0.12 03.07.2000 fix interface name handling */ /*****************************************************************************/ #include #include #include #include #include #include #include #include #include #include "sm.h" /* --------------------------------------------------------------------- */ /*static*/ const char sm_drvname[] = "soundmodem"; static const char sm_drvinfo[] = KERN_INFO "soundmodem: (C) 1996-2000 Thomas Sailer, HB9JNX/AE4WA\n" KERN_INFO "soundmodem: version 0.12 compiled " __TIME__ " " __DATE__ "\n"; /* --------------------------------------------------------------------- */ /*static*/ const struct modem_tx_info *sm_modem_tx_table[] = { #ifdef CONFIG_SOUNDMODEM_AFSK1200 &sm_afsk1200_tx, #endif /* CONFIG_SOUNDMODEM_AFSK1200 */ #ifdef CONFIG_SOUNDMODEM_AFSK2400_7 &sm_afsk2400_7_tx, #endif /* CONFIG_SOUNDMODEM_AFSK2400_7 */ #ifdef CONFIG_SOUNDMODEM_AFSK2400_8 &sm_afsk2400_8_tx, #endif /* CONFIG_SOUNDMODEM_AFSK2400_8 */ #ifdef CONFIG_SOUNDMODEM_AFSK2666 &sm_afsk2666_tx, #endif /* CONFIG_SOUNDMODEM_AFSK2666 */ #ifdef CONFIG_SOUNDMODEM_PSK4800 &sm_psk4800_tx, #endif /* CONFIG_SOUNDMODEM_PSK4800 */ #ifdef CONFIG_SOUNDMODEM_HAPN4800 &sm_hapn4800_8_tx, &sm_hapn4800_10_tx, &sm_hapn4800_pm8_tx, &sm_hapn4800_pm10_tx, #endif /* CONFIG_SOUNDMODEM_HAPN4800 */ #ifdef CONFIG_SOUNDMODEM_FSK9600 &sm_fsk9600_4_tx, &sm_fsk9600_5_tx, #endif /* CONFIG_SOUNDMODEM_FSK9600 */ NULL }; /*static*/ const struct modem_rx_info *sm_modem_rx_table[] = { #ifdef CONFIG_SOUNDMODEM_AFSK1200 &sm_afsk1200_rx, #endif /* CONFIG_SOUNDMODEM_AFSK1200 */ #ifdef CONFIG_SOUNDMODEM_AFSK2400_7 &sm_afsk2400_7_rx, #endif /* CONFIG_SOUNDMODEM_AFSK2400_7 */ #ifdef CONFIG_SOUNDMODEM_AFSK2400_8 &sm_afsk2400_8_rx, #endif /* CONFIG_SOUNDMODEM_AFSK2400_8 */ #ifdef CONFIG_SOUNDMODEM_AFSK2666 &sm_afsk2666_rx, #endif /* CONFIG_SOUNDMODEM_AFSK2666 */ #ifdef CONFIG_SOUNDMODEM_PSK4800 &sm_psk4800_rx, #endif /* CONFIG_SOUNDMODEM_PSK4800 */ #ifdef CONFIG_SOUNDMODEM_HAPN4800 &sm_hapn4800_8_rx, &sm_hapn4800_10_rx, &sm_hapn4800_pm8_rx, &sm_hapn4800_pm10_rx, #endif /* CONFIG_SOUNDMODEM_HAPN4800 */ #ifdef CONFIG_SOUNDMODEM_FSK9600 &sm_fsk9600_4_rx, &sm_fsk9600_5_rx, #endif /* CONFIG_SOUNDMODEM_FSK9600 */ NULL }; static const struct hardware_info *sm_hardware_table[] = { #ifdef CONFIG_SOUNDMODEM_SBC &sm_hw_sbc, &sm_hw_sbcfdx, #endif /* CONFIG_SOUNDMODEM_SBC */ #ifdef CONFIG_SOUNDMODEM_WSS &sm_hw_wss, &sm_hw_wssfdx, #endif /* CONFIG_SOUNDMODEM_WSS */ NULL }; /* --------------------------------------------------------------------- */ #define NR_PORTS 4 static struct net_device sm_device[NR_PORTS]; /* --------------------------------------------------------------------- */ #define UART_RBR(iobase) (iobase+0) #define UART_THR(iobase) (iobase+0) #define UART_IER(iobase) (iobase+1) #define UART_IIR(iobase) (iobase+2) #define UART_FCR(iobase) (iobase+2) #define UART_LCR(iobase) (iobase+3) #define UART_MCR(iobase) (iobase+4) #define UART_LSR(iobase) (iobase+5) #define UART_MSR(iobase) (iobase+6) #define UART_SCR(iobase) (iobase+7) #define UART_DLL(iobase) (iobase+0) #define UART_DLM(iobase) (iobase+1) #define SER_EXTENT 8 #define MIDI_DATA(iobase) (iobase) #define MIDI_STATUS(iobase) (iobase+1) #define MIDI_READ_FULL 0x80 /* attention: negative logic!! */ #define MIDI_WRITE_EMPTY 0x40 /* attention: negative logic!! */ #define MIDI_EXTENT 2 /* ---------------------------------------------------------------------- */ #define PARAM_TXDELAY 1 #define PARAM_PERSIST 2 #define PARAM_SLOTTIME 3 #define PARAM_TXTAIL 4 #define PARAM_FULLDUP 5 #define PARAM_HARDWARE 6 #define PARAM_RETURN 255 #define SP_SER 1 #define SP_PAR 2 #define SP_MIDI 4 /* * ===================== port checking routines ======================== */ enum uart { c_uart_unknown, c_uart_8250, c_uart_16450, c_uart_16550, c_uart_16550A}; static const char *uart_str[] = { "unknown", "8250", "16450", "16550", "16550A" }; static enum uart check_uart(unsigned int iobase) { unsigned char b1,b2,b3; enum uart u; enum uart uart_tab[] = { c_uart_16450, c_uart_unknown, c_uart_16550, c_uart_16550A }; if (iobase <= 0 || iobase > 0x1000-SER_EXTENT) return c_uart_unknown; if (check_region(iobase, SER_EXTENT)) return c_uart_unknown; b1 = inb(UART_MCR(iobase)); outb(b1 | 0x10, UART_MCR(iobase)); /* loopback mode */ b2 = inb(UART_MSR(iobase)); outb(0x1a, UART_MCR(iobase)); b3 = inb(UART_MSR(iobase)) & 0xf0; outb(b1, UART_MCR(iobase)); /* restore old values */ outb(b2, UART_MSR(iobase)); if (b3 != 0x90) return c_uart_unknown; inb(UART_RBR(iobase)); inb(UART_RBR(iobase)); outb(0x01, UART_FCR(iobase)); /* enable FIFOs */ u = uart_tab[(inb(UART_IIR(iobase)) >> 6) & 3]; if (u == c_uart_16450) { outb(0x5a, UART_SCR(iobase)); b1 = inb(UART_SCR(iobase)); outb(0xa5, UART_SCR(iobase)); b2 = inb(UART_SCR(iobase)); if ((b1 != 0x5a) || (b2 != 0xa5)) u = c_uart_8250; } return u; } /* --------------------------------------------------------------------- */ static int check_midi(unsigned int iobase) { unsigned long timeout; unsigned long flags; unsigned char b; if (iobase <= 0 || iobase > 0x1000-MIDI_EXTENT) return 0; if (check_region(iobase, MIDI_EXTENT)) return 0; timeout = jiffies + (HZ / 100); while (inb(MIDI_STATUS(iobase)) & MIDI_WRITE_EMPTY) if ((signed)(jiffies - timeout) > 0) return 0; save_flags(flags); cli(); outb(0xff, MIDI_DATA(iobase)); b = inb(MIDI_STATUS(iobase)); restore_flags(flags); if (!(b & MIDI_WRITE_EMPTY)) return 0; while (inb(MIDI_STATUS(iobase)) & MIDI_WRITE_EMPTY) if ((signed)(jiffies - timeout) > 0) return 0; return 1; } /* --------------------------------------------------------------------- */ void sm_output_status(struct sm_state *sm) { int invert_dcd = 0; int invert_ptt = 0; int ptt = /*hdlcdrv_ptt(&sm->hdrv)*/(sm->dma.ptt_cnt > 0) ^ invert_ptt; int dcd = (!!sm->hdrv.hdlcrx.dcd) ^ invert_dcd; if (sm->hdrv.ptt_out.flags & SP_SER) { outb(dcd | (ptt << 1), UART_MCR(sm->hdrv.ptt_out.seriobase)); outb(0x40 & (-ptt), UART_LCR(sm->hdrv.ptt_out.seriobase)); } if (sm->hdrv.ptt_out.flags & SP_PAR && sm->pardev && sm->pardev->port) parport_write_data(sm->pardev->port, ptt | (dcd << 1)); if (sm->hdrv.ptt_out.flags & SP_MIDI && hdlcdrv_ptt(&sm->hdrv)) outb(0, MIDI_DATA(sm->hdrv.ptt_out.midiiobase)); } /* --------------------------------------------------------------------- */ static void sm_output_open(struct sm_state *sm, const char *ifname) { enum uart u = c_uart_unknown; struct parport *pp = NULL; sm->hdrv.ptt_out.flags = 0; if (sm->hdrv.ptt_out.seriobase > 0 && sm->hdrv.ptt_out.seriobase <= 0x1000-SER_EXTENT && ((u = check_uart(sm->hdrv.ptt_out.seriobase))) != c_uart_unknown) { sm->hdrv.ptt_out.flags |= SP_SER; request_region(sm->hdrv.ptt_out.seriobase, SER_EXTENT, "sm ser ptt"); outb(0, UART_IER(sm->hdrv.ptt_out.seriobase)); /* 5 bits, 1 stop, no parity, no break, Div latch access */ outb(0x80, UART_LCR(sm->hdrv.ptt_out.seriobase)); outb(0, UART_DLM(sm->hdrv.ptt_out.seriobase)); outb(1, UART_DLL(sm->hdrv.ptt_out.seriobase)); /* as fast as possible */ /* LCR and MCR set by output_status */ } sm->pardev = NULL; if (sm->hdrv.ptt_out.pariobase > 0) { pp = parport_enumerate(); while (pp && pp->base != sm->hdrv.ptt_out.pariobase) pp = pp->next; if (!pp) printk(KERN_WARNING "%s: parport at address 0x%x not found\n", sm_drvname, sm->hdrv.ptt_out.pariobase); else if ((~pp->modes) & (PARPORT_MODE_PCSPP | PARPORT_MODE_SAFEININT)) printk(KERN_WARNING "%s: parport at address 0x%x cannot be used\n", sm_drvname, sm->hdrv.ptt_out.pariobase); else { sm->pardev = parport_register_device(pp, ifname, NULL, NULL, NULL, PARPORT_DEV_EXCL, NULL); if (!sm->pardev) { pp = NULL; printk(KERN_WARNING "%s: cannot register parport device (address 0x%x)\n", sm_drvname, sm->hdrv.ptt_out.pariobase); } else { if (parport_claim(sm->pardev)) { parport_unregister_device(sm->pardev); sm->pardev = NULL; printk(KERN_WARNING "%s: cannot claim parport at address 0x%x\n", sm_drvname, sm->hdrv.ptt_out.pariobase); } else sm->hdrv.ptt_out.flags |= SP_PAR; } } } if (sm->hdrv.ptt_out.midiiobase > 0 && sm->hdrv.ptt_out.midiiobase <= 0x1000-MIDI_EXTENT && check_midi(sm->hdrv.ptt_out.midiiobase)) { sm->hdrv.ptt_out.flags |= SP_MIDI; request_region(sm->hdrv.ptt_out.midiiobase, MIDI_EXTENT, "sm midi ptt"); } sm_output_status(sm); printk(KERN_INFO "%s: ptt output:", sm_drvname); if (sm->hdrv.ptt_out.flags & SP_SER) printk(" serial interface at 0x%x, uart %s", sm->hdrv.ptt_out.seriobase, uart_str[u]); if (sm->hdrv.ptt_out.flags & SP_PAR) printk(" parallel interface at 0x%x", sm->hdrv.ptt_out.pariobase); if (sm->hdrv.ptt_out.flags & SP_MIDI) printk(" mpu401 (midi) interface at 0x%x", sm->hdrv.ptt_out.midiiobase); if (!sm->hdrv.ptt_out.flags) printk(" none"); printk("\n"); } /* --------------------------------------------------------------------- */ static void sm_output_close(struct sm_state *sm) { /* release regions used for PTT output */ sm->hdrv.hdlctx.ptt = sm->hdrv.hdlctx.calibrate = 0; sm_output_status(sm); if (sm->hdrv.ptt_out.flags & SP_SER) release_region(sm->hdrv.ptt_out.seriobase, SER_EXTENT); if (sm->hdrv.ptt_out.flags & SP_PAR && sm->pardev) { parport_release(sm->pardev); parport_unregister_device(sm->pardev); } if (sm->hdrv.ptt_out.flags & SP_MIDI) release_region(sm->hdrv.ptt_out.midiiobase, MIDI_EXTENT); sm->hdrv.ptt_out.flags = 0; } /* --------------------------------------------------------------------- */ static int sm_open(struct net_device *dev); static int sm_close(struct net_device *dev); static int sm_ioctl(struct net_device *dev, struct ifreq *ifr, struct hdlcdrv_ioctl *hi, int cmd); /* --------------------------------------------------------------------- */ static const struct hdlcdrv_ops sm_ops = { sm_drvname, sm_drvinfo, sm_open, sm_close, sm_ioctl }; /* --------------------------------------------------------------------- */ static int sm_open(struct net_device *dev) { struct sm_state *sm; int err; if (!dev || !dev->priv || ((struct sm_state *)dev->priv)->hdrv.magic != HDLCDRV_MAGIC) { printk(KERN_ERR "sm_open: invalid device struct\n"); return -EINVAL; } sm = (struct sm_state *)dev->priv; if (!sm->mode_tx || !sm->mode_rx || !sm->hwdrv || !sm->hwdrv->open) return -ENODEV; sm->hdrv.par.bitrate = sm->mode_rx->bitrate; err = sm->hwdrv->open(dev, sm); if (err) return err; sm_output_open(sm, dev->name); MOD_INC_USE_COUNT; printk(KERN_INFO "%s: %s mode %s.%s at iobase 0x%lx irq %u dma %u dma2 %u\n", sm_drvname, sm->hwdrv->hw_name, sm->mode_tx->name, sm->mode_rx->name, dev->base_addr, dev->irq, dev->dma, sm->hdrv.ptt_out.dma2); return 0; } /* --------------------------------------------------------------------- */ static int sm_close(struct net_device *dev) { struct sm_state *sm; int err = -ENODEV; if (!dev || !dev->priv || ((struct sm_state *)dev->priv)->hdrv.magic != HDLCDRV_MAGIC) { printk(KERN_ERR "sm_close: invalid device struct\n"); return -EINVAL; } sm = (struct sm_state *)dev->priv; if (sm->hwdrv && sm->hwdrv->close) err = sm->hwdrv && sm->hwdrv->close(dev, sm); sm_output_close(sm); MOD_DEC_USE_COUNT; printk(KERN_INFO "%s: close %s at iobase 0x%lx irq %u dma %u\n", sm_drvname, sm->hwdrv->hw_name, dev->base_addr, dev->irq, dev->dma); return err; } /* --------------------------------------------------------------------- */ static int sethw(struct net_device *dev, struct sm_state *sm, char *mode) { char *cp = strchr(mode, ':'); const struct hardware_info **hwp = sm_hardware_table; if (!cp) cp = mode; else { *cp++ = '\0'; while (hwp && (*hwp) && (*hwp)->hw_name && strcmp((*hwp)->hw_name, mode)) hwp++; if (!hwp || !*hwp || !(*hwp)->hw_name) return -EINVAL; if ((*hwp)->loc_storage > sizeof(sm->hw)) { printk(KERN_ERR "%s: insufficient storage for hw driver %s (%d)\n", sm_drvname, (*hwp)->hw_name, (*hwp)->loc_storage); return -EINVAL; } sm->hwdrv = *hwp; } if (!*cp) return 0; if (sm->hwdrv && sm->hwdrv->sethw) return sm->hwdrv->sethw(dev, sm, cp); return -EINVAL; } /* --------------------------------------------------------------------- */ static int sm_ioctl(struct net_device *dev, struct ifreq *ifr, struct hdlcdrv_ioctl *hi, int cmd) { struct sm_state *sm; struct sm_ioctl bi; unsigned long flags; unsigned int newdiagmode; unsigned int newdiagflags; char *cp; const struct modem_tx_info **mtp = sm_modem_tx_table; const struct modem_rx_info **mrp = sm_modem_rx_table; const struct hardware_info **hwp = sm_hardware_table; if (!dev || !dev->priv || ((struct sm_state *)dev->priv)->hdrv.magic != HDLCDRV_MAGIC) { printk(KERN_ERR "sm_ioctl: invalid device struct\n"); return -EINVAL; } sm = (struct sm_state *)dev->priv; if (cmd != SIOCDEVPRIVATE) { if (!sm->hwdrv || !sm->hwdrv->ioctl) return sm->hwdrv->ioctl(dev, sm, ifr, hi, cmd); return -ENOIOCTLCMD; } switch (hi->cmd) { default: if (sm->hwdrv && sm->hwdrv->ioctl) return sm->hwdrv->ioctl(dev, sm, ifr, hi, cmd); return -ENOIOCTLCMD; case HDLCDRVCTL_GETMODE: cp = hi->data.modename; if (sm->hwdrv && sm->hwdrv->hw_name) cp += sprintf(cp, "%s:", sm->hwdrv->hw_name); else cp += sprintf(cp, ":"); if (sm->mode_tx && sm->mode_tx->name) cp += sprintf(cp, "%s", sm->mode_tx->name); else cp += sprintf(cp, ""); if (!sm->mode_rx || !sm->mode_rx || strcmp(sm->mode_rx->name, sm->mode_tx->name)) { if (sm->mode_rx && sm->mode_rx->name) cp += sprintf(cp, ",%s", sm->mode_rx->name); else cp += sprintf(cp, ","); } if (copy_to_user(ifr->ifr_data, hi, sizeof(*hi))) return -EFAULT; return 0; case HDLCDRVCTL_SETMODE: if (netif_running(dev) || !capable(CAP_NET_ADMIN)) return -EACCES; hi->data.modename[sizeof(hi->data.modename)-1] = '\0'; return sethw(dev, sm, hi->data.modename); case HDLCDRVCTL_MODELIST: cp = hi->data.modename; while (*hwp) { if ((*hwp)->hw_name) cp += sprintf(cp, "%s:,", (*hwp)->hw_name); hwp++; } while (*mtp) { if ((*mtp)->name) cp += sprintf(cp, ">%s,", (*mtp)->name); mtp++; } while (*mrp) { if ((*mrp)->name) cp += sprintf(cp, "<%s,", (*mrp)->name); mrp++; } cp[-1] = '\0'; if (copy_to_user(ifr->ifr_data, hi, sizeof(*hi))) return -EFAULT; return 0; #ifdef SM_DEBUG case SMCTL_GETDEBUG: if (copy_from_user(&bi, ifr->ifr_data, sizeof(bi))) return -EFAULT; bi.data.dbg.int_rate = sm->debug_vals.last_intcnt; bi.data.dbg.mod_cycles = sm->debug_vals.mod_cyc; bi.data.dbg.demod_cycles = sm->debug_vals.demod_cyc; bi.data.dbg.dma_residue = sm->debug_vals.dma_residue; sm->debug_vals.mod_cyc = sm->debug_vals.demod_cyc = sm->debug_vals.dma_residue = 0; if (copy_to_user(ifr->ifr_data, &bi, sizeof(bi))) return -EFAULT; return 0; #endif /* SM_DEBUG */ case SMCTL_DIAGNOSE: if (copy_from_user(&bi, ifr->ifr_data, sizeof(bi))) return -EFAULT; newdiagmode = bi.data.diag.mode; newdiagflags = bi.data.diag.flags; if (newdiagmode > SM_DIAGMODE_CONSTELLATION) return -EINVAL; bi.data.diag.mode = sm->diag.mode; bi.data.diag.flags = sm->diag.flags; bi.data.diag.samplesperbit = sm->mode_rx->sperbit; if (sm->diag.mode != newdiagmode) { save_flags(flags); cli(); sm->diag.ptr = -1; sm->diag.flags = newdiagflags & ~SM_DIAGFLAG_VALID; sm->diag.mode = newdiagmode; restore_flags(flags); if (copy_to_user(ifr->ifr_data, &bi, sizeof(bi))) return -EFAULT; return 0; } if (sm->diag.ptr < 0 || sm->diag.mode == SM_DIAGMODE_OFF) { if (copy_to_user(ifr->ifr_data, &bi, sizeof(bi))) return -EFAULT; return 0; } if (bi.data.diag.datalen > DIAGDATALEN) bi.data.diag.datalen = DIAGDATALEN; if (sm->diag.ptr < bi.data.diag.datalen) { if (copy_to_user(ifr->ifr_data, &bi, sizeof(bi))) return -EFAULT; return 0; } if (copy_to_user(bi.data.diag.data, sm->diag.data, bi.data.diag.datalen * sizeof(short))) return -EFAULT; bi.data.diag.flags |= SM_DIAGFLAG_VALID; save_flags(flags); cli(); sm->diag.ptr = -1; sm->diag.flags = newdiagflags & ~SM_DIAGFLAG_VALID; sm->diag.mode = newdiagmode; restore_flags(flags); if (copy_to_user(ifr->ifr_data, &bi, sizeof(bi))) return -EFAULT; return 0; } } /* --------------------------------------------------------------------- */ /* * command line settable parameters */ static char *mode[NR_PORTS] = { [0 ... NR_PORTS-1] = NULL }; static int iobase[NR_PORTS] = { [0 ... NR_PORTS-1] = -1 }; static int irq[NR_PORTS] = { [0 ... NR_PORTS-1] = -1 }; static int dma[NR_PORTS] = { [0 ... NR_PORTS-1] = -1 }; static int dma2[NR_PORTS] = { [0 ... NR_PORTS-1] = -1 }; static int serio[NR_PORTS] = { [0 ... NR_PORTS-1] = 0 }; static int pario[NR_PORTS] = { [0 ... NR_PORTS-1] = 0 }; static int midiio[NR_PORTS] = { [0 ... NR_PORTS-1] = 0 }; MODULE_PARM(mode, "1-" __MODULE_STRING(NR_PORTS) "s"); MODULE_PARM_DESC(mode, "soundmodem operating mode; eg. sbc:afsk1200 or wss:fsk9600"); MODULE_PARM(iobase, "1-" __MODULE_STRING(NR_PORTS) "i"); MODULE_PARM_DESC(iobase, "soundmodem base address"); MODULE_PARM(irq, "1-" __MODULE_STRING(NR_PORTS) "i"); MODULE_PARM_DESC(irq, "soundmodem interrupt"); MODULE_PARM(dma, "1-" __MODULE_STRING(NR_PORTS) "i"); MODULE_PARM_DESC(dma, "soundmodem dma channel"); MODULE_PARM(dma2, "1-" __MODULE_STRING(NR_PORTS) "i"); MODULE_PARM_DESC(dma2, "soundmodem 2nd dma channel; full duplex only"); MODULE_PARM(serio, "1-" __MODULE_STRING(NR_PORTS) "i"); MODULE_PARM_DESC(serio, "soundmodem PTT output on serial port"); MODULE_PARM(pario, "1-" __MODULE_STRING(NR_PORTS) "i"); MODULE_PARM_DESC(pario, "soundmodem PTT output on parallel port"); MODULE_PARM(midiio, "1-" __MODULE_STRING(NR_PORTS) "i"); MODULE_PARM_DESC(midiio, "soundmodem PTT output on midi port"); MODULE_AUTHOR("Thomas M. Sailer, sailer@ife.ee.ethz.ch, hb9jnx@hb9w.che.eu"); MODULE_DESCRIPTION("Soundcard amateur radio modem driver"); /* --------------------------------------------------------------------- */ static int __init init_soundmodem(void) { int i, j, found = 0; char set_hw = 1; struct sm_state *sm; printk(sm_drvinfo); /* * register net devices */ for (i = 0; i < NR_PORTS; i++) { struct net_device *dev = sm_device+i; char ifname[IFNAMSIZ]; sprintf(ifname, "sm%d", i); if (!mode[i]) set_hw = 0; else { if (!strncmp(mode[i], "sbc", 3)) { if (iobase[i] == -1) iobase[i] = 0x220; if (irq[i] == -1) irq[i] = 5; if (dma[i] == -1) dma[i] = 1; } else { if (iobase[i] == -1) iobase[i] = 0x530; if (irq[i] == -1) irq[i] = 11; if (dma[i] == -1) dma[i] = 1; } } if (!set_hw) iobase[i] = irq[i] = 0; j = hdlcdrv_register_hdlcdrv(dev, &sm_ops, sizeof(struct sm_state), ifname, iobase[i], irq[i], dma[i]); if (!j) { sm = (struct sm_state *)dev->priv; sm->hdrv.ptt_out.dma2 = dma2[i]; sm->hdrv.ptt_out.seriobase = serio[i]; sm->hdrv.ptt_out.pariobase = pario[i]; sm->hdrv.ptt_out.midiiobase = midiio[i]; if (set_hw && sethw(dev, sm, mode[i])) set_hw = 0; found++; } else { printk(KERN_WARNING "%s: cannot register net device\n", sm_drvname); } } if (!found) return -ENXIO; return 0; } static void __exit cleanup_soundmodem(void) { int i; printk(KERN_INFO "sm: cleanup_module called\n"); for(i = 0; i < NR_PORTS; i++) { struct net_device *dev = sm_device+i; struct sm_state *sm = (struct sm_state *)dev->priv; if (sm) { if (sm->hdrv.magic != HDLCDRV_MAGIC) printk(KERN_ERR "sm: invalid magic in " "cleanup_module\n"); else hdlcdrv_unregister_hdlcdrv(dev); } } } module_init(init_soundmodem); module_exit(cleanup_soundmodem); /* --------------------------------------------------------------------- */ #ifndef MODULE /* * format: soundmodem=io,irq,dma[,dma2[,serio[,pario]]],mode * mode: hw:modem * hw: sbc, wss, wssfdx * modem: afsk1200, fsk9600 */ static int __init sm_setup(char *str) { static unsigned nr_dev; int ints[8]; if (nr_dev >= NR_PORTS) return 0; str = get_options(str, 8, ints); mode[nr_dev] = str; if (ints[0] >= 1) iobase[nr_dev] = ints[1]; if (ints[0] >= 2) irq[nr_dev] = ints[2]; if (ints[0] >= 3) dma[nr_dev] = ints[3]; if (ints[0] >= 4) dma2[nr_dev] = ints[4]; if (ints[0] >= 5) serio[nr_dev] = ints[5]; if (ints[0] >= 6) pario[nr_dev] = ints[6]; if (ints[0] >= 7) midiio[nr_dev] = ints[7]; nr_dev++; return 1; } __setup("soundmodem=", sm_setup); #endif /* MODULE */ /* --------------------------------------------------------------------- */