sm.c
上传用户:jlfgdled
上传日期:2013-04-10
资源大小:33168k
文件大小:23k
源码类别:

Linux/Unix编程

开发平台:

Unix_Linux

  1. /*****************************************************************************/
  2. /*
  3.  * sm.c  -- soundcard radio modem driver.
  4.  *
  5.  * Copyright (C) 1996-2000  Thomas Sailer (sailer@ife.ee.ethz.ch)
  6.  *
  7.  * This program is free software; you can redistribute it and/or modify
  8.  * it under the terms of the GNU General Public License as published by
  9.  * the Free Software Foundation; either version 2 of the License, or
  10.  * (at your option) any later version.
  11.  *
  12.  * This program is distributed in the hope that it will be useful,
  13.  * but WITHOUT ANY WARRANTY; without even the implied warranty of
  14.  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
  15.  * GNU General Public License for more details.
  16.  *
  17.  * You should have received a copy of the GNU General Public License
  18.  * along with this program; if not, write to the Free Software
  19.  * Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
  20.  *
  21.  *  Please note that the GPL allows you to use the driver, NOT the radio.
  22.  *  In order to use the radio, you need a license from the communications
  23.  *  authority of your country.
  24.  *
  25.  *
  26.  *  Command line options (insmod command line)
  27.  *
  28.  *  mode     mode string; eg. "wss:afsk1200"
  29.  *  iobase   base address of the soundcard; common values are 0x220 for sbc,
  30.  *           0x530 for wss
  31.  *  irq      interrupt number; common values are 7 or 5 for sbc, 11 for wss
  32.  *  dma      dma number; common values are 0 or 1
  33.  *
  34.  *
  35.  *  History:
  36.  *   0.1  21.09.1996  Started
  37.  *        18.10.1996  Changed to new user space access routines (copy_{to,from}_user)
  38.  *   0.4  21.01.1997  Separately compileable soundcard/modem modules
  39.  *   0.5  03.03.1997  fixed LPT probing (check_lpt result was interpreted the wrong way round)
  40.  *   0.6  16.04.1997  init code/data tagged
  41.  *   0.7  30.07.1997  fixed halfduplex interrupt handlers/hotfix for CS423X
  42.  *   0.8  14.04.1998  cleanups
  43.  *   0.9  03.08.1999  adapt to Linus' new __setup/__initcall
  44.  *                    use parport lowlevel drivers instead of directly writing to a parallel port
  45.  *                    removed some pre-2.2 kernel compatibility cruft
  46.  *   0.10 10.08.1999  Check if parport can do SPP and is safe to access during interrupt contexts
  47.  *   0.11 12.02.2000  adapted to softnet driver interface
  48.  *   0.12 03.07.2000  fix interface name handling
  49.  */
  50. /*****************************************************************************/
  51. #include <linux/config.h>
  52. #include <linux/version.h>
  53. #include <linux/module.h>
  54. #include <linux/ioport.h>
  55. #include <linux/string.h>
  56. #include <linux/init.h>
  57. #include <linux/parport.h>
  58. #include <asm/uaccess.h>
  59. #include <asm/io.h>
  60. #include "sm.h"
  61. /* --------------------------------------------------------------------- */
  62. /*static*/ const char sm_drvname[] = "soundmodem";
  63. static const char sm_drvinfo[] = KERN_INFO "soundmodem: (C) 1996-2000 Thomas Sailer, HB9JNX/AE4WAn"
  64. KERN_INFO "soundmodem: version 0.12 compiled " __TIME__ " " __DATE__ "n";
  65. /* --------------------------------------------------------------------- */
  66. /*static*/ const struct modem_tx_info *sm_modem_tx_table[] = {
  67. #ifdef CONFIG_SOUNDMODEM_AFSK1200
  68. &sm_afsk1200_tx,
  69. #endif /* CONFIG_SOUNDMODEM_AFSK1200 */
  70. #ifdef CONFIG_SOUNDMODEM_AFSK2400_7
  71. &sm_afsk2400_7_tx,
  72. #endif /* CONFIG_SOUNDMODEM_AFSK2400_7 */
  73. #ifdef CONFIG_SOUNDMODEM_AFSK2400_8
  74. &sm_afsk2400_8_tx,
  75. #endif /* CONFIG_SOUNDMODEM_AFSK2400_8 */
  76. #ifdef CONFIG_SOUNDMODEM_AFSK2666
  77. &sm_afsk2666_tx,
  78. #endif /* CONFIG_SOUNDMODEM_AFSK2666 */
  79. #ifdef CONFIG_SOUNDMODEM_PSK4800
  80. &sm_psk4800_tx,
  81. #endif /* CONFIG_SOUNDMODEM_PSK4800 */
  82. #ifdef CONFIG_SOUNDMODEM_HAPN4800
  83. &sm_hapn4800_8_tx,
  84. &sm_hapn4800_10_tx,
  85. &sm_hapn4800_pm8_tx,
  86. &sm_hapn4800_pm10_tx,
  87. #endif /* CONFIG_SOUNDMODEM_HAPN4800 */
  88. #ifdef CONFIG_SOUNDMODEM_FSK9600
  89. &sm_fsk9600_4_tx,
  90. &sm_fsk9600_5_tx,
  91. #endif /* CONFIG_SOUNDMODEM_FSK9600 */
  92. NULL
  93. };
  94. /*static*/ const struct modem_rx_info *sm_modem_rx_table[] = {
  95. #ifdef CONFIG_SOUNDMODEM_AFSK1200
  96. &sm_afsk1200_rx,
  97. #endif /* CONFIG_SOUNDMODEM_AFSK1200 */
  98. #ifdef CONFIG_SOUNDMODEM_AFSK2400_7
  99. &sm_afsk2400_7_rx,
  100. #endif /* CONFIG_SOUNDMODEM_AFSK2400_7 */
  101. #ifdef CONFIG_SOUNDMODEM_AFSK2400_8
  102. &sm_afsk2400_8_rx,
  103. #endif /* CONFIG_SOUNDMODEM_AFSK2400_8 */
  104. #ifdef CONFIG_SOUNDMODEM_AFSK2666
  105. &sm_afsk2666_rx,
  106. #endif /* CONFIG_SOUNDMODEM_AFSK2666 */
  107. #ifdef CONFIG_SOUNDMODEM_PSK4800
  108. &sm_psk4800_rx,
  109. #endif /* CONFIG_SOUNDMODEM_PSK4800 */
  110. #ifdef CONFIG_SOUNDMODEM_HAPN4800
  111. &sm_hapn4800_8_rx,
  112. &sm_hapn4800_10_rx,
  113. &sm_hapn4800_pm8_rx,
  114. &sm_hapn4800_pm10_rx,
  115. #endif /* CONFIG_SOUNDMODEM_HAPN4800 */
  116. #ifdef CONFIG_SOUNDMODEM_FSK9600
  117. &sm_fsk9600_4_rx,
  118. &sm_fsk9600_5_rx,
  119. #endif /* CONFIG_SOUNDMODEM_FSK9600 */
  120. NULL
  121. };
  122. static const struct hardware_info *sm_hardware_table[] = {
  123. #ifdef CONFIG_SOUNDMODEM_SBC
  124. &sm_hw_sbc,
  125. &sm_hw_sbcfdx,
  126. #endif /* CONFIG_SOUNDMODEM_SBC */
  127. #ifdef CONFIG_SOUNDMODEM_WSS
  128. &sm_hw_wss,
  129. &sm_hw_wssfdx,
  130. #endif /* CONFIG_SOUNDMODEM_WSS */
  131. NULL
  132. };
  133. /* --------------------------------------------------------------------- */
  134. #define NR_PORTS 4
  135. static struct net_device sm_device[NR_PORTS];
  136. /* --------------------------------------------------------------------- */
  137. #define UART_RBR(iobase) (iobase+0)
  138. #define UART_THR(iobase) (iobase+0)
  139. #define UART_IER(iobase) (iobase+1)
  140. #define UART_IIR(iobase) (iobase+2)
  141. #define UART_FCR(iobase) (iobase+2)
  142. #define UART_LCR(iobase) (iobase+3)
  143. #define UART_MCR(iobase) (iobase+4)
  144. #define UART_LSR(iobase) (iobase+5)
  145. #define UART_MSR(iobase) (iobase+6)
  146. #define UART_SCR(iobase) (iobase+7)
  147. #define UART_DLL(iobase) (iobase+0)
  148. #define UART_DLM(iobase) (iobase+1)
  149. #define SER_EXTENT 8
  150. #define MIDI_DATA(iobase)     (iobase)
  151. #define MIDI_STATUS(iobase)   (iobase+1)
  152. #define MIDI_READ_FULL 0x80   /* attention: negative logic!! */
  153. #define MIDI_WRITE_EMPTY 0x40 /* attention: negative logic!! */
  154. #define MIDI_EXTENT 2
  155. /* ---------------------------------------------------------------------- */
  156. #define PARAM_TXDELAY   1
  157. #define PARAM_PERSIST   2
  158. #define PARAM_SLOTTIME  3
  159. #define PARAM_TXTAIL    4
  160. #define PARAM_FULLDUP   5
  161. #define PARAM_HARDWARE  6
  162. #define PARAM_RETURN    255
  163. #define SP_SER  1
  164. #define SP_PAR  2
  165. #define SP_MIDI 4
  166. /*
  167.  * ===================== port checking routines ========================
  168.  */
  169. enum uart { c_uart_unknown, c_uart_8250,
  170. c_uart_16450, c_uart_16550, c_uart_16550A};
  171. static const char *uart_str[] =
  172. { "unknown", "8250", "16450", "16550", "16550A" };
  173. static enum uart check_uart(unsigned int iobase)
  174. {
  175. unsigned char b1,b2,b3;
  176. enum uart u;
  177. enum uart uart_tab[] =
  178. { c_uart_16450, c_uart_unknown, c_uart_16550, c_uart_16550A };
  179. if (iobase <= 0 || iobase > 0x1000-SER_EXTENT)
  180. return c_uart_unknown;
  181. if (check_region(iobase, SER_EXTENT))
  182. return c_uart_unknown;
  183. b1 = inb(UART_MCR(iobase));
  184. outb(b1 | 0x10, UART_MCR(iobase)); /* loopback mode */
  185. b2 = inb(UART_MSR(iobase));
  186. outb(0x1a, UART_MCR(iobase));
  187. b3 = inb(UART_MSR(iobase)) & 0xf0;
  188. outb(b1, UART_MCR(iobase));    /* restore old values */
  189. outb(b2, UART_MSR(iobase));
  190. if (b3 != 0x90)
  191. return c_uart_unknown;
  192. inb(UART_RBR(iobase));
  193. inb(UART_RBR(iobase));
  194. outb(0x01, UART_FCR(iobase)); /* enable FIFOs */
  195. u = uart_tab[(inb(UART_IIR(iobase)) >> 6) & 3];
  196. if (u == c_uart_16450) {
  197. outb(0x5a, UART_SCR(iobase));
  198. b1 = inb(UART_SCR(iobase));
  199. outb(0xa5, UART_SCR(iobase));
  200. b2 = inb(UART_SCR(iobase));
  201. if ((b1 != 0x5a) || (b2 != 0xa5))
  202. u = c_uart_8250;
  203. }
  204. return u;
  205. }
  206. /* --------------------------------------------------------------------- */
  207. static int check_midi(unsigned int iobase)
  208. {
  209. unsigned long timeout;
  210. unsigned long flags;
  211. unsigned char b;
  212. if (iobase <= 0 || iobase > 0x1000-MIDI_EXTENT)
  213. return 0;
  214. if (check_region(iobase, MIDI_EXTENT))
  215. return 0;
  216. timeout = jiffies + (HZ / 100);
  217. while (inb(MIDI_STATUS(iobase)) & MIDI_WRITE_EMPTY)
  218. if ((signed)(jiffies - timeout) > 0)
  219. return 0;
  220. save_flags(flags);
  221. cli();
  222. outb(0xff, MIDI_DATA(iobase));
  223. b = inb(MIDI_STATUS(iobase));
  224. restore_flags(flags);
  225. if (!(b & MIDI_WRITE_EMPTY))
  226. return 0;
  227. while (inb(MIDI_STATUS(iobase)) & MIDI_WRITE_EMPTY)
  228. if ((signed)(jiffies - timeout) > 0)
  229. return 0;
  230. return 1;
  231. }
  232. /* --------------------------------------------------------------------- */
  233. void sm_output_status(struct sm_state *sm)
  234. {
  235. int invert_dcd = 0;
  236. int invert_ptt = 0;
  237. int ptt = /*hdlcdrv_ptt(&sm->hdrv)*/(sm->dma.ptt_cnt > 0) ^ invert_ptt;
  238. int dcd = (!!sm->hdrv.hdlcrx.dcd) ^ invert_dcd;
  239. if (sm->hdrv.ptt_out.flags & SP_SER) {
  240. outb(dcd | (ptt << 1), UART_MCR(sm->hdrv.ptt_out.seriobase));
  241. outb(0x40 & (-ptt), UART_LCR(sm->hdrv.ptt_out.seriobase));
  242. }
  243. if (sm->hdrv.ptt_out.flags & SP_PAR && sm->pardev && sm->pardev->port)
  244. parport_write_data(sm->pardev->port, ptt | (dcd << 1));
  245. if (sm->hdrv.ptt_out.flags & SP_MIDI && hdlcdrv_ptt(&sm->hdrv))
  246. outb(0, MIDI_DATA(sm->hdrv.ptt_out.midiiobase));
  247. }
  248. /* --------------------------------------------------------------------- */
  249. static void sm_output_open(struct sm_state *sm, const char *ifname)
  250. {
  251. enum uart u = c_uart_unknown;
  252. struct parport *pp = NULL;
  253. sm->hdrv.ptt_out.flags = 0;
  254. if (sm->hdrv.ptt_out.seriobase > 0 &&
  255.     sm->hdrv.ptt_out.seriobase <= 0x1000-SER_EXTENT &&
  256.     ((u = check_uart(sm->hdrv.ptt_out.seriobase))) != c_uart_unknown) {
  257. sm->hdrv.ptt_out.flags |= SP_SER;
  258. request_region(sm->hdrv.ptt_out.seriobase, SER_EXTENT, "sm ser ptt");
  259. outb(0, UART_IER(sm->hdrv.ptt_out.seriobase));
  260. /* 5 bits, 1 stop, no parity, no break, Div latch access */
  261. outb(0x80, UART_LCR(sm->hdrv.ptt_out.seriobase));
  262. outb(0, UART_DLM(sm->hdrv.ptt_out.seriobase));
  263. outb(1, UART_DLL(sm->hdrv.ptt_out.seriobase)); /* as fast as possible */
  264. /* LCR and MCR set by output_status */
  265. }
  266. sm->pardev = NULL;
  267. if (sm->hdrv.ptt_out.pariobase > 0) {
  268. pp = parport_enumerate();
  269. while (pp && pp->base != sm->hdrv.ptt_out.pariobase) 
  270. pp = pp->next;
  271. if (!pp)
  272. printk(KERN_WARNING "%s: parport at address 0x%x not foundn", sm_drvname, sm->hdrv.ptt_out.pariobase);
  273. else if ((~pp->modes) & (PARPORT_MODE_PCSPP | PARPORT_MODE_SAFEININT))
  274. printk(KERN_WARNING "%s: parport at address 0x%x cannot be usedn", sm_drvname, sm->hdrv.ptt_out.pariobase);
  275. else {
  276. sm->pardev = parport_register_device(pp, ifname, NULL, NULL, NULL, PARPORT_DEV_EXCL, NULL);
  277. if (!sm->pardev) {
  278. pp = NULL;
  279. printk(KERN_WARNING "%s: cannot register parport device (address 0x%x)n", sm_drvname, sm->hdrv.ptt_out.pariobase);
  280. } else {
  281. if (parport_claim(sm->pardev)) {
  282. parport_unregister_device(sm->pardev);
  283. sm->pardev = NULL;
  284. printk(KERN_WARNING "%s: cannot claim parport at address 0x%xn", sm_drvname, sm->hdrv.ptt_out.pariobase);
  285. } else
  286. sm->hdrv.ptt_out.flags |= SP_PAR;
  287. }
  288. }
  289. }
  290. if (sm->hdrv.ptt_out.midiiobase > 0 &&
  291.     sm->hdrv.ptt_out.midiiobase <= 0x1000-MIDI_EXTENT &&
  292.     check_midi(sm->hdrv.ptt_out.midiiobase)) {
  293. sm->hdrv.ptt_out.flags |= SP_MIDI;
  294. request_region(sm->hdrv.ptt_out.midiiobase, MIDI_EXTENT,
  295.        "sm midi ptt");
  296. }
  297. sm_output_status(sm);
  298. printk(KERN_INFO "%s: ptt output:", sm_drvname);
  299. if (sm->hdrv.ptt_out.flags & SP_SER)
  300. printk(" serial interface at 0x%x, uart %s", sm->hdrv.ptt_out.seriobase,
  301.        uart_str[u]);
  302. if (sm->hdrv.ptt_out.flags & SP_PAR)
  303. printk(" parallel interface at 0x%x", sm->hdrv.ptt_out.pariobase);
  304. if (sm->hdrv.ptt_out.flags & SP_MIDI)
  305. printk(" mpu401 (midi) interface at 0x%x", sm->hdrv.ptt_out.midiiobase);
  306. if (!sm->hdrv.ptt_out.flags)
  307. printk(" none");
  308. printk("n");
  309. }
  310. /* --------------------------------------------------------------------- */
  311. static void sm_output_close(struct sm_state *sm)
  312. {
  313. /* release regions used for PTT output */
  314. sm->hdrv.hdlctx.ptt = sm->hdrv.hdlctx.calibrate = 0;
  315. sm_output_status(sm);
  316. if (sm->hdrv.ptt_out.flags & SP_SER)
  317. release_region(sm->hdrv.ptt_out.seriobase, SER_EXTENT);
  318.         if (sm->hdrv.ptt_out.flags & SP_PAR && sm->pardev) {
  319.         parport_release(sm->pardev);
  320. parport_unregister_device(sm->pardev);
  321. }
  322.         if (sm->hdrv.ptt_out.flags & SP_MIDI)
  323. release_region(sm->hdrv.ptt_out.midiiobase, MIDI_EXTENT);
  324. sm->hdrv.ptt_out.flags = 0;
  325. }
  326. /* --------------------------------------------------------------------- */
  327. static int sm_open(struct net_device *dev);
  328. static int sm_close(struct net_device *dev);
  329. static int sm_ioctl(struct net_device *dev, struct ifreq *ifr,
  330.     struct hdlcdrv_ioctl *hi, int cmd);
  331. /* --------------------------------------------------------------------- */
  332. static const struct hdlcdrv_ops sm_ops = {
  333. sm_drvname, sm_drvinfo, sm_open, sm_close, sm_ioctl
  334. };
  335. /* --------------------------------------------------------------------- */
  336. static int sm_open(struct net_device *dev)
  337. {
  338. struct sm_state *sm;
  339. int err;
  340. if (!dev || !dev->priv ||
  341.     ((struct sm_state *)dev->priv)->hdrv.magic != HDLCDRV_MAGIC) {
  342. printk(KERN_ERR "sm_open: invalid device structn");
  343. return -EINVAL;
  344. }
  345. sm = (struct sm_state *)dev->priv;
  346. if (!sm->mode_tx || !sm->mode_rx || !sm->hwdrv || !sm->hwdrv->open)
  347. return -ENODEV;
  348. sm->hdrv.par.bitrate = sm->mode_rx->bitrate;
  349. err = sm->hwdrv->open(dev, sm);
  350. if (err)
  351. return err;
  352. sm_output_open(sm, dev->name);
  353. MOD_INC_USE_COUNT;
  354. printk(KERN_INFO "%s: %s mode %s.%s at iobase 0x%lx irq %u dma %u dma2 %un",
  355.        sm_drvname, sm->hwdrv->hw_name, sm->mode_tx->name,
  356.        sm->mode_rx->name, dev->base_addr, dev->irq, dev->dma, sm->hdrv.ptt_out.dma2);
  357. return 0;
  358. }
  359. /* --------------------------------------------------------------------- */
  360. static int sm_close(struct net_device *dev)
  361. {
  362. struct sm_state *sm;
  363. int err = -ENODEV;
  364. if (!dev || !dev->priv ||
  365.     ((struct sm_state *)dev->priv)->hdrv.magic != HDLCDRV_MAGIC) {
  366. printk(KERN_ERR "sm_close: invalid device structn");
  367. return -EINVAL;
  368. }
  369. sm = (struct sm_state *)dev->priv;
  370. if (sm->hwdrv && sm->hwdrv->close)
  371. err = sm->hwdrv && sm->hwdrv->close(dev, sm);
  372. sm_output_close(sm);
  373. MOD_DEC_USE_COUNT;
  374. printk(KERN_INFO "%s: close %s at iobase 0x%lx irq %u dma %un",
  375.        sm_drvname, sm->hwdrv->hw_name, dev->base_addr, dev->irq, dev->dma);
  376. return err;
  377. }
  378. /* --------------------------------------------------------------------- */
  379. static int sethw(struct net_device *dev, struct sm_state *sm, char *mode)
  380. {
  381. char *cp = strchr(mode, ':');
  382. const struct hardware_info **hwp = sm_hardware_table;
  383. if (!cp)
  384. cp = mode;
  385. else {
  386. *cp++ = '';
  387. while (hwp && (*hwp) && (*hwp)->hw_name && strcmp((*hwp)->hw_name, mode))
  388. hwp++;
  389. if (!hwp || !*hwp || !(*hwp)->hw_name)
  390. return -EINVAL;
  391. if ((*hwp)->loc_storage > sizeof(sm->hw)) {
  392. printk(KERN_ERR "%s: insufficient storage for hw driver %s (%d)n",
  393.        sm_drvname, (*hwp)->hw_name, (*hwp)->loc_storage);
  394. return -EINVAL;
  395. }
  396. sm->hwdrv = *hwp;
  397. }
  398. if (!*cp)
  399. return 0;
  400. if (sm->hwdrv && sm->hwdrv->sethw)
  401. return sm->hwdrv->sethw(dev, sm, cp);
  402. return -EINVAL;
  403. }
  404. /* --------------------------------------------------------------------- */
  405. static int sm_ioctl(struct net_device *dev, struct ifreq *ifr,
  406.     struct hdlcdrv_ioctl *hi, int cmd)
  407. {
  408. struct sm_state *sm;
  409. struct sm_ioctl bi;
  410. unsigned long flags;
  411. unsigned int newdiagmode;
  412. unsigned int newdiagflags;
  413. char *cp;
  414. const struct modem_tx_info **mtp = sm_modem_tx_table;
  415. const struct modem_rx_info **mrp = sm_modem_rx_table;
  416. const struct hardware_info **hwp = sm_hardware_table;
  417. if (!dev || !dev->priv ||
  418.     ((struct sm_state *)dev->priv)->hdrv.magic != HDLCDRV_MAGIC) {
  419. printk(KERN_ERR "sm_ioctl: invalid device structn");
  420. return -EINVAL;
  421. }
  422. sm = (struct sm_state *)dev->priv;
  423. if (cmd != SIOCDEVPRIVATE) {
  424. if (!sm->hwdrv || !sm->hwdrv->ioctl)
  425. return sm->hwdrv->ioctl(dev, sm, ifr, hi, cmd);
  426. return -ENOIOCTLCMD;
  427. }
  428. switch (hi->cmd) {
  429. default:
  430. if (sm->hwdrv && sm->hwdrv->ioctl)
  431. return sm->hwdrv->ioctl(dev, sm, ifr, hi, cmd);
  432. return -ENOIOCTLCMD;
  433. case HDLCDRVCTL_GETMODE:
  434. cp = hi->data.modename;
  435. if (sm->hwdrv && sm->hwdrv->hw_name)
  436. cp += sprintf(cp, "%s:", sm->hwdrv->hw_name);
  437. else
  438. cp += sprintf(cp, "<unspec>:");
  439. if (sm->mode_tx && sm->mode_tx->name)
  440. cp += sprintf(cp, "%s", sm->mode_tx->name);
  441. else
  442. cp += sprintf(cp, "<unspec>");
  443. if (!sm->mode_rx || !sm->mode_rx ||
  444.     strcmp(sm->mode_rx->name, sm->mode_tx->name)) {
  445. if (sm->mode_rx && sm->mode_rx->name)
  446. cp += sprintf(cp, ",%s", sm->mode_rx->name);
  447. else
  448. cp += sprintf(cp, ",<unspec>");
  449. }
  450. if (copy_to_user(ifr->ifr_data, hi, sizeof(*hi)))
  451. return -EFAULT;
  452. return 0;
  453. case HDLCDRVCTL_SETMODE:
  454. if (netif_running(dev) || !capable(CAP_NET_ADMIN))
  455. return -EACCES;
  456. hi->data.modename[sizeof(hi->data.modename)-1] = '';
  457. return sethw(dev, sm, hi->data.modename);
  458. case HDLCDRVCTL_MODELIST:
  459. cp = hi->data.modename;
  460. while (*hwp) {
  461. if ((*hwp)->hw_name)
  462. cp += sprintf(cp, "%s:,", (*hwp)->hw_name);
  463. hwp++;
  464. }
  465. while (*mtp) {
  466. if ((*mtp)->name)
  467. cp += sprintf(cp, ">%s,", (*mtp)->name);
  468. mtp++;
  469. }
  470. while (*mrp) {
  471. if ((*mrp)->name)
  472. cp += sprintf(cp, "<%s,", (*mrp)->name);
  473. mrp++;
  474. }
  475. cp[-1] = '';
  476. if (copy_to_user(ifr->ifr_data, hi, sizeof(*hi)))
  477. return -EFAULT;
  478. return 0;
  479. #ifdef SM_DEBUG
  480. case SMCTL_GETDEBUG:
  481. if (copy_from_user(&bi, ifr->ifr_data, sizeof(bi)))
  482. return -EFAULT;
  483. bi.data.dbg.int_rate = sm->debug_vals.last_intcnt;
  484. bi.data.dbg.mod_cycles = sm->debug_vals.mod_cyc;
  485. bi.data.dbg.demod_cycles = sm->debug_vals.demod_cyc;
  486. bi.data.dbg.dma_residue = sm->debug_vals.dma_residue;
  487. sm->debug_vals.mod_cyc = sm->debug_vals.demod_cyc =
  488. sm->debug_vals.dma_residue = 0;
  489. if (copy_to_user(ifr->ifr_data, &bi, sizeof(bi)))
  490. return -EFAULT;
  491. return 0;
  492. #endif /* SM_DEBUG */
  493. case SMCTL_DIAGNOSE:
  494. if (copy_from_user(&bi, ifr->ifr_data, sizeof(bi)))
  495. return -EFAULT;
  496. newdiagmode = bi.data.diag.mode;
  497. newdiagflags = bi.data.diag.flags;
  498. if (newdiagmode > SM_DIAGMODE_CONSTELLATION)
  499. return -EINVAL;
  500. bi.data.diag.mode = sm->diag.mode;
  501. bi.data.diag.flags = sm->diag.flags;
  502. bi.data.diag.samplesperbit = sm->mode_rx->sperbit;
  503. if (sm->diag.mode != newdiagmode) {
  504. save_flags(flags);
  505. cli();
  506. sm->diag.ptr = -1;
  507. sm->diag.flags = newdiagflags & ~SM_DIAGFLAG_VALID;
  508. sm->diag.mode = newdiagmode;
  509. restore_flags(flags);
  510. if (copy_to_user(ifr->ifr_data, &bi, sizeof(bi)))
  511. return -EFAULT;
  512. return 0;
  513. }
  514. if (sm->diag.ptr < 0 || sm->diag.mode == SM_DIAGMODE_OFF) {
  515. if (copy_to_user(ifr->ifr_data, &bi, sizeof(bi)))
  516. return -EFAULT;
  517. return 0;
  518. }
  519. if (bi.data.diag.datalen > DIAGDATALEN)
  520. bi.data.diag.datalen = DIAGDATALEN;
  521. if (sm->diag.ptr < bi.data.diag.datalen) {
  522. if (copy_to_user(ifr->ifr_data, &bi, sizeof(bi)))
  523. return -EFAULT;
  524. return 0;
  525. }
  526. if (copy_to_user(bi.data.diag.data, sm->diag.data,
  527.  bi.data.diag.datalen * sizeof(short)))
  528. return -EFAULT;
  529. bi.data.diag.flags |= SM_DIAGFLAG_VALID;
  530. save_flags(flags);
  531. cli();
  532. sm->diag.ptr = -1;
  533. sm->diag.flags = newdiagflags & ~SM_DIAGFLAG_VALID;
  534. sm->diag.mode = newdiagmode;
  535. restore_flags(flags);
  536. if (copy_to_user(ifr->ifr_data, &bi, sizeof(bi)))
  537. return -EFAULT;
  538. return 0;
  539. }
  540. }
  541. /* --------------------------------------------------------------------- */
  542. /*
  543.  * command line settable parameters
  544.  */
  545. static char *mode[NR_PORTS] = { [0 ... NR_PORTS-1] = NULL };
  546. static int iobase[NR_PORTS] = { [0 ... NR_PORTS-1] = -1 };
  547. static int irq[NR_PORTS] = { [0 ... NR_PORTS-1] = -1 };
  548. static int dma[NR_PORTS] = { [0 ... NR_PORTS-1] = -1 };
  549. static int dma2[NR_PORTS] = { [0 ... NR_PORTS-1] = -1 };
  550. static int serio[NR_PORTS] = { [0 ... NR_PORTS-1] = 0 };
  551. static int pario[NR_PORTS] = { [0 ... NR_PORTS-1] = 0 };
  552. static int midiio[NR_PORTS] = { [0 ... NR_PORTS-1] = 0 };
  553. MODULE_PARM(mode, "1-" __MODULE_STRING(NR_PORTS) "s");
  554. MODULE_PARM_DESC(mode, "soundmodem operating mode; eg. sbc:afsk1200 or wss:fsk9600");
  555. MODULE_PARM(iobase, "1-" __MODULE_STRING(NR_PORTS) "i");
  556. MODULE_PARM_DESC(iobase, "soundmodem base address");
  557. MODULE_PARM(irq, "1-" __MODULE_STRING(NR_PORTS) "i");
  558. MODULE_PARM_DESC(irq, "soundmodem interrupt");
  559. MODULE_PARM(dma, "1-" __MODULE_STRING(NR_PORTS) "i");
  560. MODULE_PARM_DESC(dma, "soundmodem dma channel");
  561. MODULE_PARM(dma2, "1-" __MODULE_STRING(NR_PORTS) "i");
  562. MODULE_PARM_DESC(dma2, "soundmodem 2nd dma channel; full duplex only");
  563. MODULE_PARM(serio, "1-" __MODULE_STRING(NR_PORTS) "i");
  564. MODULE_PARM_DESC(serio, "soundmodem PTT output on serial port");
  565. MODULE_PARM(pario, "1-" __MODULE_STRING(NR_PORTS) "i");
  566. MODULE_PARM_DESC(pario, "soundmodem PTT output on parallel port");
  567. MODULE_PARM(midiio, "1-" __MODULE_STRING(NR_PORTS) "i");
  568. MODULE_PARM_DESC(midiio, "soundmodem PTT output on midi port");
  569. MODULE_AUTHOR("Thomas M. Sailer, sailer@ife.ee.ethz.ch, hb9jnx@hb9w.che.eu");
  570. MODULE_DESCRIPTION("Soundcard amateur radio modem driver");
  571. /* --------------------------------------------------------------------- */
  572. static int __init init_soundmodem(void)
  573. {
  574. int i, j, found = 0;
  575. char set_hw = 1;
  576. struct sm_state *sm;
  577. printk(sm_drvinfo);
  578. /*
  579.  * register net devices
  580.  */
  581. for (i = 0; i < NR_PORTS; i++) {
  582. struct net_device *dev = sm_device+i;
  583. char ifname[IFNAMSIZ];
  584. sprintf(ifname, "sm%d", i);
  585. if (!mode[i])
  586. set_hw = 0;
  587. else {
  588. if (!strncmp(mode[i], "sbc", 3)) {
  589. if (iobase[i] == -1)
  590. iobase[i] = 0x220;
  591. if (irq[i] == -1)
  592. irq[i] = 5;
  593. if (dma[i] == -1)
  594. dma[i] = 1;
  595. } else {
  596. if (iobase[i] == -1)
  597. iobase[i] = 0x530;
  598. if (irq[i] == -1)
  599. irq[i] = 11;
  600. if (dma[i] == -1)
  601. dma[i] = 1;
  602. }
  603. }
  604. if (!set_hw)
  605. iobase[i] = irq[i] = 0;
  606. j = hdlcdrv_register_hdlcdrv(dev, &sm_ops, sizeof(struct sm_state), ifname, iobase[i], irq[i], dma[i]);
  607. if (!j) {
  608. sm = (struct sm_state *)dev->priv;
  609. sm->hdrv.ptt_out.dma2 = dma2[i];
  610. sm->hdrv.ptt_out.seriobase = serio[i];
  611. sm->hdrv.ptt_out.pariobase = pario[i];
  612. sm->hdrv.ptt_out.midiiobase = midiio[i];
  613. if (set_hw && sethw(dev, sm, mode[i]))
  614. set_hw = 0;
  615. found++;
  616. } else {
  617. printk(KERN_WARNING "%s: cannot register net devicen", sm_drvname);
  618. }
  619. }
  620. if (!found)
  621. return -ENXIO;
  622. return 0;
  623. }
  624. static void __exit cleanup_soundmodem(void)
  625. {
  626. int i;
  627. printk(KERN_INFO "sm: cleanup_module calledn");
  628. for(i = 0; i < NR_PORTS; i++) {
  629. struct net_device *dev = sm_device+i;
  630. struct sm_state *sm = (struct sm_state *)dev->priv;
  631. if (sm) {
  632. if (sm->hdrv.magic != HDLCDRV_MAGIC)
  633. printk(KERN_ERR "sm: invalid magic in "
  634.        "cleanup_modulen");
  635. else
  636. hdlcdrv_unregister_hdlcdrv(dev);
  637. }
  638. }
  639. }
  640. module_init(init_soundmodem);
  641. module_exit(cleanup_soundmodem);
  642. /* --------------------------------------------------------------------- */
  643. #ifndef MODULE
  644. /*
  645.  * format: soundmodem=io,irq,dma[,dma2[,serio[,pario]]],mode
  646.  * mode: hw:modem
  647.  * hw: sbc, wss, wssfdx
  648.  * modem: afsk1200, fsk9600
  649.  */
  650. static int __init sm_setup(char *str)
  651. {
  652. static unsigned nr_dev;
  653. int ints[8];
  654. if (nr_dev >= NR_PORTS)
  655. return 0;
  656. str = get_options(str, 8, ints);
  657. mode[nr_dev] = str;
  658. if (ints[0] >= 1)
  659. iobase[nr_dev] = ints[1];
  660. if (ints[0] >= 2)
  661. irq[nr_dev] = ints[2];
  662. if (ints[0] >= 3)
  663. dma[nr_dev] = ints[3];
  664. if (ints[0] >= 4)
  665. dma2[nr_dev] = ints[4];
  666. if (ints[0] >= 5)
  667. serio[nr_dev] = ints[5];
  668. if (ints[0] >= 6)
  669. pario[nr_dev] = ints[6];
  670. if (ints[0] >= 7)
  671. midiio[nr_dev] = ints[7];
  672. nr_dev++;
  673. return 1;
  674. }
  675. __setup("soundmodem=", sm_setup);
  676. #endif /* MODULE */
  677. /* --------------------------------------------------------------------- */