!C99Shell v. 1.0 pre-release build #13!

Software: Apache/2.0.54 (Unix) mod_perl/1.99_09 Perl/v5.8.0 mod_ssl/2.0.54 OpenSSL/0.9.7l DAV/2 FrontPage/5.0.2.2635 PHP/4.4.0 mod_gzip/2.0.26.1a 

uname -a: Linux snow.he.net 4.4.276-v2-mono-1 #1 SMP Wed Jul 21 11:21:17 PDT 2021 i686 

uid=99(nobody) gid=98(nobody) groups=98(nobody) 

Safe-mode: OFF (not secure)

/usr/src/linux-2.4.18-xfs-1.1/drivers/net/hamradio/soundmodem/   drwxr-xr-x
Free 318.34 GB of 458.09 GB (69.49%)
Home    Back    Forward    UPDIR    Refresh    Search    Buffer    Encoder    Tools    Proc.    FTP brute    Sec.    SQL    PHP-code    Update    Feedback    Self remove    Logout    


Viewing file:     sm.c (22.74 KB)      -rw-r--r--
Select action/file-type:
(+) | (+) | (+) | Code (+) | Session (+) | (+) | SDB (+) | (+) | (+) | (+) | (+) | (+) |
/*****************************************************************************/

/*
 *    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 <linux/config.h>
#include <linux/version.h>
#include <linux/module.h>
#include <linux/ioport.h>
#include <linux/string.h>
#include <linux/init.h>
#include <linux/parport.h>
#include <asm/uaccess.h>
#include <asm/io.h>
#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, "<unspec>:");
        if (sm->mode_tx && sm->mode_tx->name)
            cp += sprintf(cp, "%s", sm->mode_tx->name);
        else
            cp += sprintf(cp, "<unspec>");
        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, ",<unspec>");
        }
        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 */
/* --------------------------------------------------------------------- */

:: Command execute ::

Enter:
 
Select:
 

:: Search ::
  - regexp 

:: Upload ::
 
[ Read-Only ]

:: Make Dir ::
 
[ Read-Only ]
:: Make File ::
 
[ Read-Only ]

:: Go Dir ::
 
:: Go File ::
 

--[ c99shell v. 1.0 pre-release build #13 powered by Captain Crunch Security Team | http://ccteam.ru | Generation time: 0.0089 ]--