amd7930.c
上传用户:lgb322
上传日期:2013-02-24
资源大小:30529k
文件大小:56k
源码类别:

嵌入式Linux

开发平台:

Unix_Linux

  1. /* $Id: amd7930.c,v 1.28 2001/10/13 01:47:29 davem Exp $
  2.  * drivers/sbus/audio/amd7930.c
  3.  *
  4.  * Copyright (C) 1996,1997 Thomas K. Dyas (tdyas@eden.rutgers.edu)
  5.  *
  6.  * This is the lowlevel driver for the AMD7930 audio chip found on all
  7.  * sun4c machines and some sun4m machines.
  8.  *
  9.  * The amd7930 is actually an ISDN chip which has a very simple
  10.  * integrated audio encoder/decoder. When Sun decided on what chip to
  11.  * use for audio, they had the brilliant idea of using the amd7930 and
  12.  * only connecting the audio encoder/decoder pins.
  13.  *
  14.  * Thanks to the AMD engineer who was able to get us the AMD79C30
  15.  * databook which has all the programming information and gain tables.
  16.  *
  17.  * Advanced Micro Devices' Am79C30A is an ISDN/audio chip used in the
  18.  * SparcStation 1+.  The chip provides microphone and speaker interfaces
  19.  * which provide mono-channel audio at 8K samples per second via either
  20.  * 8-bit A-law or 8-bit mu-law encoding.  Also, the chip features an
  21.  * ISDN BRI Line Interface Unit (LIU), I.430 S/T physical interface,
  22.  * which performs basic D channel LAPD processing and provides raw
  23.  * B channel data.  The digital audio channel, the two ISDN B channels,
  24.  * and two 64 Kbps channels to the microprocessor are all interconnected
  25.  * via a multiplexer.
  26.  *
  27.  * This driver interfaces to the Linux HiSax ISDN driver, which performs
  28.  * all high-level Q.921 and Q.931 ISDN functions.  The file is not
  29.  * itself a hardware driver; rather it uses functions exported by
  30.  * the AMD7930 driver in the sparcaudio subsystem (drivers/sbus/audio),
  31.  * allowing the chip to be simultaneously used for both audio and ISDN data.
  32.  * The hardware driver does _no_ buffering, but provides several callbacks
  33.  * which are called during interrupt service and should therefore run quickly.
  34.  *
  35.  * D channel transmission is performed by passing the hardware driver the
  36.  * address and size of an skb's data area, then waiting for a callback
  37.  * to signal successful transmission of the packet.  A task is then
  38.  * queued to notify the HiSax driver that another packet may be transmitted.
  39.  *
  40.  * D channel reception is quite simple, mainly because of:
  41.  *   1) the slow speed of the D channel - 16 kbps, and
  42.  *   2) the presence of an 8- or 32-byte (depending on chip version) FIFO
  43.  *      to buffer the D channel data on the chip
  44.  * Worst case scenario of back-to-back packets with the 8 byte buffer
  45.  * at 16 kbps yields an service time of 4 ms - long enough to preclude
  46.  * the need for fancy buffering.  We queue a background task that copies
  47.  * data out of the receive buffer into an skb, and the hardware driver
  48.  * simply does nothing until we're done with the receive buffer and
  49.  * reset it for a new packet.
  50.  *
  51.  * B channel processing is more complex, because of:
  52.  *   1) the faster speed - 64 kbps,
  53.  *   2) the lack of any on-chip buffering (it interrupts for every byte), and
  54.  *   3) the lack of any chip support for HDLC encapsulation
  55.  *
  56.  * The HiSax driver can put each B channel into one of three modes -
  57.  * L1_MODE_NULL (channel disabled), L1_MODE_TRANS (transparent data relay),
  58.  * and L1_MODE_HDLC (HDLC encapsulation by low-level driver).
  59.  * L1_MODE_HDLC is the most common, used for almost all "pure" digital
  60.  * data sessions.  L1_MODE_TRANS is used for ISDN audio.
  61.  *
  62.  * HDLC B channel transmission is performed via a large buffer into
  63.  * which the skb is copied while performing HDLC bit-stuffing.  A CRC
  64.  * is computed and attached to the end of the buffer, which is then
  65.  * passed to the low-level routines for raw transmission.  Once
  66.  * transmission is complete, the hardware driver is set to enter HDLC
  67.  * idle by successive transmission of mark (all 1) bytes, waiting for
  68.  * the ISDN driver to prepare another packet for transmission and
  69.  * deliver it.
  70.  *
  71.  * HDLC B channel reception is performed via an X-byte ring buffer
  72.  * divided into N sections of X/N bytes each.  Defaults: X=256 bytes, N=4.
  73.  * As the hardware driver notifies us that each section is full, we
  74.  * hand it the next section and schedule a background task to peruse
  75.  * the received section, bit-by-bit, with an HDLC decoder.  As
  76.  * packets are detected, they are copied into a large buffer while
  77.  * decoding HDLC bit-stuffing.  The ending CRC is verified, and if
  78.  * it is correct, we alloc a new skb of the correct length (which we
  79.  * now know), copy the packet into it, and hand it to the upper layers.
  80.  * Optimization: for large packets, we hand the buffer (which also
  81.  * happens to be an skb) directly to the upper layer after an skb_trim,
  82.  * and alloc a new large buffer for future packets, thus avoiding a copy.
  83.  * Then we return to HDLC processing; state is saved between calls.
  84.  */
  85. #include <linux/module.h>
  86. #include <linux/kernel.h>
  87. #include <linux/sched.h>
  88. #include <linux/errno.h>
  89. #include <linux/interrupt.h>
  90. #include <linux/slab.h>
  91. #include <linux/init.h>
  92. #include <linux/version.h>
  93. #include <linux/soundcard.h>
  94. #include <asm/openprom.h>
  95. #include <asm/oplib.h>
  96. #include <asm/system.h>
  97. #include <asm/irq.h>
  98. #include <asm/io.h>
  99. #include <asm/sbus.h>
  100. #include <asm/audioio.h>
  101. #include "amd7930.h"
  102. static __u8  bilinear2mulaw(__u8 data);
  103. static __u8  mulaw2bilinear(__u8 data);
  104. static __u8  linear2mulaw(__u16 data);
  105. static __u16 mulaw2linear(__u8 data);
  106. #if defined (AMD79C30_ISDN)
  107. #include "../../isdn/hisax/hisax.h"
  108. #include "../../isdn/hisax/isdnl1.h"
  109. #include "../../isdn/hisax/foreign.h"
  110. #endif
  111. #define MAX_DRIVERS 1
  112. static struct sparcaudio_driver drivers[MAX_DRIVERS];
  113. static int num_drivers;
  114. /* Each amd7930 chip has two bi-directional B channels and a D
  115.  * channel available to the uproc.  This structure handles all
  116.  * the buffering needed to transmit and receive via a single channel.
  117.  */
  118. #define CHANNEL_AVAILABLE 0x00
  119. #define CHANNEL_INUSE_AUDIO_IN 0x01
  120. #define CHANNEL_INUSE_AUDIO_OUT 0x02
  121. #define CHANNEL_INUSE_ISDN_B1 0x04
  122. #define CHANNEL_INUSE_ISDN_B2 0x08
  123. #define CHANNEL_INUSE 0xff
  124. struct amd7930_channel {
  125. /* Channel status */
  126. u8 channel_status;
  127. /* Current buffer that the driver is playing on channel */
  128. volatile __u8 * output_ptr;
  129. volatile u32 output_count;
  130. u8 xmit_idle_char;
  131. /* Callback routine (and argument) when output is done on */
  132. void (*output_callback)(void *, unsigned char);
  133. void * output_callback_arg;
  134. /* Current buffer that the driver is recording on channel */
  135. volatile __u8 * input_ptr;
  136. volatile u32 input_count;
  137. volatile u32 input_limit;
  138. /* Callback routine (and argument) when input is done on */
  139. void (*input_callback)(void *, unsigned char, unsigned long);
  140. void * input_callback_arg;
  141. int input_format;
  142. int output_format;
  143. };
  144. /* Private information we store for each amd7930 chip. */
  145. struct amd7930_info {
  146. struct amd7930_channel D;
  147. struct amd7930_channel Bb;
  148. struct amd7930_channel Bc;
  149. /* Pointers to which B channels are being used for what
  150.  * These three fields (Baudio, Bisdn[0], and Bisdn[1]) will either
  151.  * be NULL or point to one of the Bb/Bc structures above.
  152.  */
  153. struct amd7930_channel *Baudio;
  154. struct amd7930_channel *Bisdn[2];
  155. /* Device registers information. */
  156. unsigned long regs;
  157. unsigned long regs_size;
  158. struct amd7930_map map;
  159. /* Volume information. */
  160. int pgain, rgain, mgain;
  161. /* Device interrupt information. */
  162. int irq;
  163. volatile int ints_on;
  164. /* Format type */
  165. int format_type;
  166. /* Someone to signal when the ISDN LIU state changes */
  167. int liu_state;
  168. void (*liu_callback)(void *);
  169. void *liu_callback_arg;
  170. };
  171. /* Output a 16-bit quantity in the order that the amd7930 expects. */
  172. static __inline__ void amd7930_out16(unsigned long regs, u16 val)
  173. {
  174. sbus_writeb(val & 0xff, regs + DR);
  175. sbus_writeb(val >> 8, regs + DR);
  176. }
  177. /* gx, gr & stg gains.  this table must contain 256 elements with
  178.  * the 0th being "infinity" (the magic value 9008).  The remaining
  179.  * elements match sun's gain curve (but with higher resolution):
  180.  * -18 to 0dB in .16dB steps then 0 to 12dB in .08dB steps.
  181.  */
  182. static __const__ __u16 gx_coeff[256] = {
  183. 0x9008, 0x8b7c, 0x8b51, 0x8b45, 0x8b42, 0x8b3b, 0x8b36, 0x8b33,
  184. 0x8b32, 0x8b2a, 0x8b2b, 0x8b2c, 0x8b25, 0x8b23, 0x8b22, 0x8b22,
  185. 0x9122, 0x8b1a, 0x8aa3, 0x8aa3, 0x8b1c, 0x8aa6, 0x912d, 0x912b,
  186. 0x8aab, 0x8b12, 0x8aaa, 0x8ab2, 0x9132, 0x8ab4, 0x913c, 0x8abb,
  187. 0x9142, 0x9144, 0x9151, 0x8ad5, 0x8aeb, 0x8a79, 0x8a5a, 0x8a4a,
  188. 0x8b03, 0x91c2, 0x91bb, 0x8a3f, 0x8a33, 0x91b2, 0x9212, 0x9213,
  189. 0x8a2c, 0x921d, 0x8a23, 0x921a, 0x9222, 0x9223, 0x922d, 0x9231,
  190. 0x9234, 0x9242, 0x925b, 0x92dd, 0x92c1, 0x92b3, 0x92ab, 0x92a4,
  191. 0x92a2, 0x932b, 0x9341, 0x93d3, 0x93b2, 0x93a2, 0x943c, 0x94b2,
  192. 0x953a, 0x9653, 0x9782, 0x9e21, 0x9d23, 0x9cd2, 0x9c23, 0x9baa,
  193. 0x9bde, 0x9b33, 0x9b22, 0x9b1d, 0x9ab2, 0xa142, 0xa1e5, 0x9a3b,
  194. 0xa213, 0xa1a2, 0xa231, 0xa2eb, 0xa313, 0xa334, 0xa421, 0xa54b,
  195. 0xada4, 0xac23, 0xab3b, 0xaaab, 0xaa5c, 0xb1a3, 0xb2ca, 0xb3bd,
  196. 0xbe24, 0xbb2b, 0xba33, 0xc32b, 0xcb5a, 0xd2a2, 0xe31d, 0x0808,
  197. 0x72ba, 0x62c2, 0x5c32, 0x52db, 0x513e, 0x4cce, 0x43b2, 0x4243,
  198. 0x41b4, 0x3b12, 0x3bc3, 0x3df2, 0x34bd, 0x3334, 0x32c2, 0x3224,
  199. 0x31aa, 0x2a7b, 0x2aaa, 0x2b23, 0x2bba, 0x2c42, 0x2e23, 0x25bb,
  200. 0x242b, 0x240f, 0x231a, 0x22bb, 0x2241, 0x2223, 0x221f, 0x1a33,
  201. 0x1a4a, 0x1acd, 0x2132, 0x1b1b, 0x1b2c, 0x1b62, 0x1c12, 0x1c32,
  202. 0x1d1b, 0x1e71, 0x16b1, 0x1522, 0x1434, 0x1412, 0x1352, 0x1323,
  203. 0x1315, 0x12bc, 0x127a, 0x1235, 0x1226, 0x11a2, 0x1216, 0x0a2a,
  204. 0x11bc, 0x11d1, 0x1163, 0x0ac2, 0x0ab2, 0x0aab, 0x0b1b, 0x0b23,
  205. 0x0b33, 0x0c0f, 0x0bb3, 0x0c1b, 0x0c3e, 0x0cb1, 0x0d4c, 0x0ec1,
  206. 0x079a, 0x0614, 0x0521, 0x047c, 0x0422, 0x03b1, 0x03e3, 0x0333,
  207. 0x0322, 0x031c, 0x02aa, 0x02ba, 0x02f2, 0x0242, 0x0232, 0x0227,
  208. 0x0222, 0x021b, 0x01ad, 0x0212, 0x01b2, 0x01bb, 0x01cb, 0x01f6,
  209. 0x0152, 0x013a, 0x0133, 0x0131, 0x012c, 0x0123, 0x0122, 0x00a2,
  210. 0x011b, 0x011e, 0x0114, 0x00b1, 0x00aa, 0x00b3, 0x00bd, 0x00ba,
  211. 0x00c5, 0x00d3, 0x00f3, 0x0062, 0x0051, 0x0042, 0x003b, 0x0033,
  212. 0x0032, 0x002a, 0x002c, 0x0025, 0x0023, 0x0022, 0x001a, 0x0021,
  213. 0x001b, 0x001b, 0x001d, 0x0015, 0x0013, 0x0013, 0x0012, 0x0012,
  214. 0x000a, 0x000a, 0x0011, 0x0011, 0x000b, 0x000b, 0x000c, 0x000e,
  215. };
  216. static __const__ __u16 ger_coeff[] = {
  217. 0x431f, /* 5. dB */
  218. 0x331f, /* 5.5 dB */
  219. 0x40dd, /* 6. dB */
  220. 0x11dd, /* 6.5 dB */
  221. 0x440f, /* 7. dB */
  222. 0x411f, /* 7.5 dB */
  223. 0x311f, /* 8. dB */
  224. 0x5520, /* 8.5 dB */
  225. 0x10dd, /* 9. dB */
  226. 0x4211, /* 9.5 dB */
  227. 0x410f, /* 10. dB */
  228. 0x111f, /* 10.5 dB */
  229. 0x600b, /* 11. dB */
  230. 0x00dd, /* 11.5 dB */
  231. 0x4210, /* 12. dB */
  232. 0x110f, /* 13. dB */
  233. 0x7200, /* 14. dB */
  234. 0x2110, /* 15. dB */
  235. 0x2200, /* 15.9 dB */
  236. 0x000b, /* 16.9 dB */
  237. 0x000f  /* 18. dB */
  238. };
  239. #define NR_GER_COEFFS (sizeof(ger_coeff) / sizeof(ger_coeff[0]))
  240. /* Enable amd7930 interrupts atomically. */
  241. static void amd7930_enable_ints(struct amd7930_info *info)
  242. {
  243. unsigned long flags;
  244. save_and_cli(flags);
  245. if (!info->ints_on) {
  246. sbus_writeb(AMR_INIT, info->regs + CR);
  247. sbus_writeb(AM_INIT_ACTIVE, info->regs + DR);
  248. info->ints_on = 1;
  249. }
  250. restore_flags(flags);
  251. }
  252. /* Disable amd7930 interrupts atomically. */
  253. static __inline__ void amd7930_disable_ints(struct amd7930_info *info)
  254. {
  255. unsigned long flags;
  256. save_and_cli(flags);
  257. if (info->ints_on) {
  258. sbus_writeb(AMR_INIT, info->regs + CR);
  259. sbus_writeb(AM_INIT_ACTIVE | AM_INIT_DISABLE_INTS,
  260.     info->regs + DR);
  261. info->ints_on = 0;
  262. }
  263. restore_flags(flags);
  264. }  
  265. /* Idle amd7930 (no interrupts, no audio, no data) */
  266. static __inline__ void amd7930_idle(struct amd7930_info *info)
  267. {
  268. unsigned long flags;
  269. save_and_cli(flags);
  270. if (info->ints_on) {
  271. sbus_writeb(AMR_INIT, info->regs + CR);
  272. sbus_writeb(0, info->regs + DR);
  273. info->ints_on = 0;
  274. }
  275. restore_flags(flags);
  276. }  
  277. /* Commit the local copy of the MAP registers to the amd7930. */
  278. static void amd7930_write_map(struct sparcaudio_driver *drv)
  279. {
  280. struct amd7930_info *info = (struct amd7930_info *) drv->private;
  281. unsigned long        regs = info->regs;
  282. struct amd7930_map  *map  = &info->map;
  283. unsigned long flags;
  284. save_and_cli(flags);
  285. sbus_writeb(AMR_MAP_GX, regs + CR);
  286. amd7930_out16(regs, map->gx);
  287. sbus_writeb(AMR_MAP_GR, regs + CR);
  288. amd7930_out16(regs, map->gr);
  289. sbus_writeb(AMR_MAP_STGR, regs + CR);
  290. amd7930_out16(regs, map->stgr);
  291. sbus_writeb(AMR_MAP_GER, regs + CR);
  292. amd7930_out16(regs, map->ger);
  293. sbus_writeb(AMR_MAP_MMR1, regs + CR);
  294. sbus_writeb(map->mmr1, regs + DR);
  295. sbus_writeb(AMR_MAP_MMR2, regs + CR);
  296. sbus_writeb(map->mmr2, regs + DR);
  297. restore_flags(flags);
  298. }
  299. /* Update the MAP registers with new settings. */
  300. static void amd7930_update_map(struct sparcaudio_driver *drv)
  301. {
  302. struct amd7930_info *info = (struct amd7930_info *) drv->private;
  303. struct amd7930_map  *map  = &info->map;
  304. int level;
  305. map->gx = gx_coeff[info->rgain];
  306. map->stgr = gx_coeff[info->mgain];
  307. level = (info->pgain * (256 + NR_GER_COEFFS)) >> 8;
  308. if (level >= 256) {
  309. map->ger = ger_coeff[level - 256];
  310. map->gr = gx_coeff[255];
  311. } else {
  312. map->ger = ger_coeff[0];
  313. map->gr = gx_coeff[level];
  314. }
  315. amd7930_write_map(drv);
  316. }
  317. /* Bit of a hack here - if the HISAX ISDN driver has got INTSTAT debugging
  318.  * turned on, we send debugging characters to the ISDN driver:
  319.  *
  320.  *   i# - Interrupt received - number from 0 to 7 is low three bits of IR
  321.  *   >  - Loaded a single char into the Dchan xmit FIFO
  322.  *   +  - Finished loading an xmit packet into the Dchan xmit FIFO
  323.  *   <  - Read a single char from the Dchan recv FIFO
  324.  *   !  - Finished reading a packet from the Dchan recv FIFO
  325.  *
  326.  * This code needs to be removed if anything other than HISAX uses the ISDN
  327.  * driver, since D.output_callback_arg is assumed to be a certain struct ptr
  328.  */
  329. #ifdef L2FRAME_DEBUG
  330. inline void debug_info(struct amd7930_info *info, char c)
  331. {
  332. struct IsdnCardState *cs;
  333. if (!info || !info->D.output_callback_arg)
  334. return;
  335. cs = (struct IsdnCardState *) info->D.output_callback_arg;
  336. if (!cs || !cs->status_write)
  337. return;
  338. if (cs->debug & L1_DEB_INTSTAT) {
  339. *(cs->status_write++) = c;
  340. if (cs->status_write > cs->status_end)
  341. cs->status_write = cs->status_buf;
  342. }
  343. }
  344. #else
  345. #define debug_info(info,c)
  346. #endif
  347. static void fill_D_xmit_fifo(struct amd7930_info *info)
  348. {
  349. /* Send next byte(s) of outgoing data. */
  350. while (info->D.output_ptr && info->D.output_count > 0 &&
  351.                (sbus_readb(info->regs + DSR2) & AMR_DSR2_TBE)) {
  352. u8 byte = *(info->D.output_ptr);
  353. /* Send the next byte and advance buffer pointer. */
  354. sbus_writeb(byte, info->regs + DCTB);
  355. info->D.output_ptr++;
  356. info->D.output_count--;
  357. debug_info(info, '>');
  358.         }
  359. }
  360. static void transceive_Dchannel(struct amd7930_info *info)
  361. {
  362. __u8 dummy;
  363. #define D_XMIT_ERRORS (AMR_DER_COLLISION | AMR_DER_UNRN)
  364. #define D_RECV_ERRORS (AMR_DER_RABRT | AMR_DER_RFRAME | AMR_DER_FCS | 
  365. AMR_DER_OVFL | AMR_DER_UNFL | AMR_DER_OVRN)
  366. /* Transmit if we can */
  367. fill_D_xmit_fifo(info);
  368. /* Done with the xmit buffer? Notify the midlevel driver. */
  369. if (info->D.output_ptr != NULL && info->D.output_count == 0) {
  370. info->D.output_ptr = NULL;
  371. info->D.output_count = 0;
  372. debug_info(info, '+');
  373. if (info->D.output_callback)
  374. (*info->D.output_callback)
  375. (info->D.output_callback_arg,
  376.  sbus_readb(info->regs + DER));
  377.  /* sbus_readb(info->regs + DER) & D_XMIT_ERRORS); */
  378. }
  379. /* Read the next byte(s) of incoming data. */
  380. while (sbus_readb(info->regs + DSR2) & AMR_DSR2_RBA) {
  381. if (info->D.input_ptr &&
  382.     (info->D.input_count < info->D.input_limit)) {
  383. /* Get the next byte and advance buffer pointer. */
  384. *(info->D.input_ptr) = sbus_readb(info->regs + DCRB);
  385. info->D.input_ptr++;
  386. info->D.input_count++;
  387. } else {
  388. /* Overflow - should be detected by chip via RBLR
  389.  * so we'll just consume data until we see LBRP
  390.  */
  391. dummy = sbus_readb(info->regs + DCRB);
  392. }
  393. debug_info(info, '<');
  394. if (sbus_readb(info->regs + DSR2) & AMR_DSR2_LBRP) {
  395. __u8 der;
  396. /* End of recv packet? Notify the midlevel driver. */
  397. debug_info(info, '!');
  398. info->D.input_ptr = NULL;
  399. der = sbus_readb(info->regs + DER) & D_RECV_ERRORS;
  400. /* Read receive byte count - advances FIFOs */
  401. sbus_writeb(AMR_DLC_DRCR, info->regs + CR);
  402. dummy = sbus_readb(info->regs + DR);
  403. dummy = sbus_readb(info->regs + DR);
  404. if (info->D.input_callback)
  405. (*info->D.input_callback)
  406. (info->D.input_callback_arg, der,
  407.  info->D.input_count);
  408. }
  409. }
  410. }
  411. long amd7930_xmit_idles = 0;
  412. static void transceive_Bchannel(struct amd7930_channel *channel,
  413. unsigned long reg)
  414. {
  415. /* Send the next byte of outgoing data. */
  416. if (channel->output_ptr && channel->output_count > 0) {
  417. u8 byte;
  418. /* Send the next byte and advance buffer pointer. */
  419. switch(channel->output_format) {
  420. case AUDIO_ENCODING_ULAW:
  421. case AUDIO_ENCODING_ALAW:
  422. byte = *(channel->output_ptr);
  423. sbus_writeb(byte, reg);
  424. break;
  425. case AUDIO_ENCODING_LINEAR8:
  426. byte = bilinear2mulaw(*(channel->output_ptr));
  427. sbus_writeb(byte, reg);
  428. break;
  429. case AUDIO_ENCODING_LINEAR:
  430. if (channel->output_count >= 2) {
  431. u16 val = channel->output_ptr[0] << 8;
  432. val |= channel->output_ptr[1];
  433. byte = linear2mulaw(val);
  434. sbus_writeb(byte, reg);
  435. channel->output_ptr++;
  436. channel->output_count--;
  437. };
  438. };
  439. channel->output_ptr++;
  440. channel->output_count--;
  441. /* Done with the buffer? Notify the midlevel driver. */
  442. if (channel->output_count == 0) {
  443. channel->output_ptr = NULL;
  444. channel->output_count = 0;
  445. if (channel->output_callback)
  446. (*channel->output_callback)
  447. (channel->output_callback_arg,1);
  448. }
  449. } else {
  450. sbus_writeb(channel->xmit_idle_char, reg);
  451. amd7930_xmit_idles++;
  452.         }
  453. /* Read the next byte of incoming data. */
  454. if (channel->input_ptr && channel->input_count > 0) {
  455. /* Get the next byte and advance buffer pointer. */
  456. switch(channel->input_format) {
  457. case AUDIO_ENCODING_ULAW:
  458. case AUDIO_ENCODING_ALAW:
  459. *(channel->input_ptr) = sbus_readb(reg);
  460. break;
  461. case AUDIO_ENCODING_LINEAR8:
  462. *(channel->input_ptr) = mulaw2bilinear(sbus_readb(reg));
  463. break;
  464. case AUDIO_ENCODING_LINEAR:
  465. if (channel->input_count >= 2) {
  466. u16 val = mulaw2linear(sbus_readb(reg));
  467. channel->input_ptr[0] = val >> 8;
  468. channel->input_ptr[1] = val & 0xff;
  469. channel->input_ptr++;
  470. channel->input_count--;
  471. } else {
  472. *(channel->input_ptr) = 0;
  473. }
  474. };
  475. channel->input_ptr++;
  476. channel->input_count--;
  477. /* Done with the buffer? Notify the midlevel driver. */
  478. if (channel->input_count == 0) {
  479. channel->input_ptr = NULL;
  480. channel->input_count = 0;
  481. if (channel->input_callback)
  482. (*channel->input_callback)
  483. (channel->input_callback_arg, 1, 0);
  484. }
  485. }
  486. }
  487. /* Interrupt handler (The chip takes only one byte per interrupt. Grrr!) */
  488. static void amd7930_interrupt(int irq, void *dev_id, struct pt_regs *intr_regs)
  489. {
  490. struct sparcaudio_driver *drv = (struct sparcaudio_driver *) dev_id;
  491. struct amd7930_info *info = (struct amd7930_info *) drv->private;
  492. unsigned long regs = info->regs;
  493. __u8 ir;
  494. /* Clear the interrupt. */
  495. ir = sbus_readb(regs + IR);
  496. if (ir & AMR_IR_BBUF) {
  497. if (info->Bb.channel_status == CHANNEL_INUSE)
  498. transceive_Bchannel(&info->Bb, info->regs + BBTB);
  499. if (info->Bc.channel_status == CHANNEL_INUSE)
  500. transceive_Bchannel(&info->Bc, info->regs + BCTB);
  501. }
  502. if (ir & (AMR_IR_DRTHRSH | AMR_IR_DTTHRSH | AMR_IR_DSRI)) {
  503. debug_info(info, 'i');
  504. debug_info(info, '0' + (ir&7));
  505. transceive_Dchannel(info);
  506. }
  507. if (ir & AMR_IR_LSRI) {
  508. __u8 lsr;
  509. sbus_writeb(AMR_LIU_LSR, regs + CR);
  510. lsr = sbus_readb(regs + DR);
  511.                 info->liu_state = (lsr & 0x7) + 2;
  512.                 if (info->liu_callback)
  513. (*info->liu_callback)(info->liu_callback_arg);
  514.         }
  515. }
  516. static int amd7930_open(struct inode * inode, struct file * file,
  517. struct sparcaudio_driver *drv)
  518. {
  519. struct amd7930_info *info = (struct amd7930_info *) drv->private;
  520. switch(MINOR(inode->i_rdev) & 0xf) {
  521. case SPARCAUDIO_AUDIO_MINOR:
  522. info->format_type = AUDIO_ENCODING_ULAW;
  523. break;
  524. case SPARCAUDIO_DSP_MINOR:
  525. info->format_type = AUDIO_ENCODING_LINEAR8;
  526. break;
  527. case SPARCAUDIO_DSP16_MINOR:
  528. info->format_type = AUDIO_ENCODING_LINEAR;
  529. break;
  530. };
  531. MOD_INC_USE_COUNT;
  532. return 0;
  533. }
  534. static void amd7930_release(struct inode * inode, struct file * file,
  535.     struct sparcaudio_driver *drv)
  536. {
  537. /* amd7930_disable_ints(drv->private); */
  538. MOD_DEC_USE_COUNT;
  539. }
  540. static void request_Baudio(struct amd7930_info *info)
  541. {
  542. if (info->Bb.channel_status == CHANNEL_AVAILABLE) {
  543. info->Bb.channel_status = CHANNEL_INUSE;
  544. info->Baudio = &info->Bb;
  545. /* Multiplexor map - audio (Ba) to Bb */
  546. sbus_writeb(AMR_MUX_MCR1, info->regs + CR);
  547. sbus_writeb(AM_MUX_CHANNEL_Ba | (AM_MUX_CHANNEL_Bb << 4),
  548.     info->regs + DR);
  549. /* Enable B channel interrupts */
  550. sbus_writeb(AMR_MUX_MCR4, info->regs + CR);
  551. sbus_writeb(AM_MUX_MCR4_ENABLE_INTS, info->regs + DR);
  552. } else if (info->Bc.channel_status == CHANNEL_AVAILABLE) {
  553. info->Bc.channel_status = CHANNEL_INUSE;
  554. info->Baudio = &info->Bc;
  555. /* Multiplexor map - audio (Ba) to Bc */
  556. sbus_writeb(AMR_MUX_MCR1, info->regs + CR);
  557. sbus_writeb(AM_MUX_CHANNEL_Ba | (AM_MUX_CHANNEL_Bc << 4),
  558.     info->regs + DR);
  559. /* Enable B channel interrupts */
  560. sbus_writeb(AMR_MUX_MCR4, info->regs + CR);
  561. sbus_writeb(AM_MUX_MCR4_ENABLE_INTS, info->regs + DR);
  562. }
  563. }
  564. static void release_Baudio(struct amd7930_info *info)
  565. {
  566. if (info->Baudio) {
  567. info->Baudio->channel_status = CHANNEL_AVAILABLE;
  568. sbus_writeb(AMR_MUX_MCR1, info->regs + CR);
  569. sbus_writeb(0, info->regs + DR);
  570. info->Baudio = NULL;
  571. if (info->Bb.channel_status == CHANNEL_AVAILABLE &&
  572.     info->Bc.channel_status == CHANNEL_AVAILABLE) {
  573. /* Disable B channel interrupts */
  574. sbus_writeb(AMR_MUX_MCR4, info->regs + CR);
  575. sbus_writeb(0, info->regs + DR);
  576. }
  577. }
  578. }
  579. static void amd7930_start_output(struct sparcaudio_driver *drv,
  580.  __u8 * buffer, unsigned long count)
  581. {
  582. struct amd7930_info *info = (struct amd7930_info *) drv->private;
  583. if (! info->Baudio)
  584. request_Baudio(info);
  585. if (info->Baudio) {
  586. info->Baudio->output_ptr = buffer;
  587. info->Baudio->output_count = count;
  588. info->Baudio->output_format = info->format_type;
  589. info->Baudio->output_callback = (void *) &sparcaudio_output_done;
  590. info->Baudio->output_callback_arg = (void *) drv;
  591. info->Baudio->xmit_idle_char = 0;
  592. }
  593. }
  594. static void amd7930_stop_output(struct sparcaudio_driver *drv)
  595. {
  596. struct amd7930_info *info = (struct amd7930_info *) drv->private;
  597. if (info->Baudio) {
  598. info->Baudio->output_ptr = NULL;
  599. info->Baudio->output_count = 0;
  600. if (! info->Baudio->input_ptr)
  601. release_Baudio(info);
  602. }
  603. }
  604. static void amd7930_start_input(struct sparcaudio_driver *drv,
  605. __u8 * buffer, unsigned long count)
  606. {
  607. struct amd7930_info *info = (struct amd7930_info *) drv->private;
  608. if (! info->Baudio)
  609. request_Baudio(info);
  610. if (info->Baudio) {
  611. info->Baudio->input_ptr = buffer;
  612. info->Baudio->input_count = count;
  613. info->Baudio->input_format = info->format_type;
  614. info->Baudio->input_callback = (void *) &sparcaudio_input_done;
  615. info->Baudio->input_callback_arg = (void *) drv;
  616. }
  617. }
  618. static void amd7930_stop_input(struct sparcaudio_driver *drv)
  619. {
  620. struct amd7930_info *info = (struct amd7930_info *) drv->private;
  621. if (info->Baudio) {
  622. info->Baudio->input_ptr = NULL;
  623. info->Baudio->input_count = 0;
  624. if (! info->Baudio->output_ptr)
  625. release_Baudio(info);
  626. }
  627. }
  628. static void amd7930_sunaudio_getdev(struct sparcaudio_driver *drv,
  629.  audio_device_t * audinfo)
  630. {
  631. strncpy(audinfo->name, "SUNW,am79c30", sizeof(audinfo->name) - 1);
  632. strncpy(audinfo->version, "a", sizeof(audinfo->version) - 1);
  633. strncpy(audinfo->config, "onboard1", sizeof(audinfo->config) - 1);
  634. }
  635. static int amd7930_sunaudio_getdev_sunos(struct sparcaudio_driver *drv)
  636. {
  637. return AUDIO_DEV_AMD;
  638. }
  639. static int amd7930_get_formats(struct sparcaudio_driver *drv)
  640. {
  641.       return (AFMT_MU_LAW | AFMT_A_LAW | AFMT_U8 | AFMT_S16_BE);
  642. }
  643. static int amd7930_get_output_ports(struct sparcaudio_driver *drv)
  644. {
  645.       return (AUDIO_SPEAKER | AUDIO_HEADPHONE);
  646. }
  647. static int amd7930_get_input_ports(struct sparcaudio_driver *drv)
  648. {
  649.       return (AUDIO_MICROPHONE);
  650. }
  651. static int amd7930_set_output_volume(struct sparcaudio_driver *drv, int vol)
  652. {
  653. struct amd7930_info *info = (struct amd7930_info *) drv->private;
  654. info->pgain = vol;
  655. amd7930_update_map(drv);
  656. return 0;
  657. }
  658. static int amd7930_get_output_volume(struct sparcaudio_driver *drv)
  659. {
  660. struct amd7930_info *info = (struct amd7930_info *) drv->private;
  661. return info->pgain;
  662. }
  663. static int amd7930_set_input_volume(struct sparcaudio_driver *drv, int vol)
  664. {
  665. struct amd7930_info *info = (struct amd7930_info *) drv->private;
  666. info->rgain = vol;
  667. amd7930_update_map(drv);
  668. return 0;
  669. }
  670. static int amd7930_get_input_volume(struct sparcaudio_driver *drv)
  671. {
  672. struct amd7930_info *info = (struct amd7930_info *) drv->private;
  673. return info->rgain;
  674. }
  675. static int amd7930_set_monitor_volume(struct sparcaudio_driver *drv, int vol)
  676. {
  677. struct amd7930_info *info = (struct amd7930_info *) drv->private;
  678. info->mgain = vol;
  679. amd7930_update_map(drv);
  680. return 0;
  681. }
  682. static int amd7930_get_monitor_volume(struct sparcaudio_driver *drv)
  683. {
  684.       struct amd7930_info *info = (struct amd7930_info *) drv->private;
  685.       return info->mgain;
  686. }
  687. /* Cheats. The amd has the minimum capabilities we support */
  688. static int amd7930_get_output_balance(struct sparcaudio_driver *drv)
  689. {
  690. return AUDIO_MID_BALANCE;
  691. }
  692. static int amd7930_get_input_balance(struct sparcaudio_driver *drv)
  693. {
  694. return AUDIO_MID_BALANCE;
  695. }
  696. static int amd7930_get_output_channels(struct sparcaudio_driver *drv)
  697. {
  698. return AUDIO_MIN_PLAY_CHANNELS;
  699. }
  700. static int amd7930_set_output_channels(struct sparcaudio_driver *drv, 
  701.        int value)
  702. {
  703.   return (value == AUDIO_MIN_PLAY_CHANNELS) ? 0 : -EINVAL;
  704. }
  705. static int amd7930_get_input_channels(struct sparcaudio_driver *drv)
  706. {
  707. return AUDIO_MIN_REC_CHANNELS;
  708. }
  709. static int 
  710. amd7930_set_input_channels(struct sparcaudio_driver *drv, int value)
  711. {
  712. return (value == AUDIO_MIN_REC_CHANNELS) ? 0 : -EINVAL;
  713. }
  714. static int amd7930_get_output_precision(struct sparcaudio_driver *drv)
  715. {
  716. return AUDIO_MIN_PLAY_PRECISION;
  717. }
  718. static int 
  719. amd7930_set_output_precision(struct sparcaudio_driver *drv, int value)
  720. {
  721. return (value == AUDIO_MIN_PLAY_PRECISION) ? 0 : -EINVAL;
  722. }
  723. static int amd7930_get_input_precision(struct sparcaudio_driver *drv)
  724. {
  725. return AUDIO_MIN_REC_PRECISION;
  726. }
  727. static int 
  728. amd7930_set_input_precision(struct sparcaudio_driver *drv, int value)
  729. {
  730. return (value == AUDIO_MIN_REC_PRECISION) ? 0 : -EINVAL;
  731. }
  732. static int amd7930_get_output_port(struct sparcaudio_driver *drv)
  733. {
  734. struct amd7930_info *info = (struct amd7930_info *) drv->private;
  735. if (info->map.mmr2 & AM_MAP_MMR2_LS)
  736. return AUDIO_SPEAKER; 
  737. return AUDIO_HEADPHONE;
  738. }
  739. static int amd7930_set_output_port(struct sparcaudio_driver *drv, int value)
  740. {
  741. struct amd7930_info *info = (struct amd7930_info *) drv->private;
  742. switch (value) {
  743. case AUDIO_HEADPHONE:
  744. info->map.mmr2 &= ~AM_MAP_MMR2_LS;
  745. break;
  746. case AUDIO_SPEAKER:
  747. info->map.mmr2 |= AM_MAP_MMR2_LS;
  748. break;
  749. default:
  750. return -EINVAL;
  751. };
  752. amd7930_update_map(drv);
  753. return 0;
  754. }
  755. /* Only a microphone here, so no troubles */
  756. static int amd7930_get_input_port(struct sparcaudio_driver *drv)
  757. {
  758. return AUDIO_MICROPHONE;
  759. }
  760. static int amd7930_get_encoding(struct sparcaudio_driver *drv)
  761. {
  762. struct amd7930_info *info = (struct amd7930_info *) drv->private;
  763. if ((info->map.mmr1 & AM_MAP_MMR1_ALAW) && 
  764.     (info->format_type == AUDIO_ENCODING_ALAW))
  765. return AUDIO_ENCODING_ALAW;
  766. return info->format_type;
  767. }
  768. static int 
  769. amd7930_set_encoding(struct sparcaudio_driver *drv, int value)
  770. {
  771. struct amd7930_info *info = (struct amd7930_info *) drv->private;
  772. switch (value) {
  773. case AUDIO_ENCODING_ALAW:
  774. info->map.mmr1 |= AM_MAP_MMR1_ALAW;
  775. break;
  776. case AUDIO_ENCODING_LINEAR8:
  777. case AUDIO_ENCODING_LINEAR:
  778. case AUDIO_ENCODING_ULAW:
  779. info->map.mmr1 &= ~AM_MAP_MMR1_ALAW;
  780. break;
  781. default:
  782. return -EINVAL;
  783. };
  784. info->format_type = value;
  785. amd7930_update_map(drv);
  786. return 0;
  787. }
  788. /* This is what you get. Take it or leave it */
  789. static int amd7930_get_output_rate(struct sparcaudio_driver *drv)
  790. {
  791. return AMD7930_RATE;
  792. }
  793. static int 
  794. amd7930_set_output_rate(struct sparcaudio_driver *drv, int value)
  795. {
  796. return (value == AMD7930_RATE) ? 0 : -EINVAL;
  797. }
  798. static int amd7930_get_input_rate(struct sparcaudio_driver *drv)
  799. {
  800. return AMD7930_RATE;
  801. }
  802. static int
  803. amd7930_set_input_rate(struct sparcaudio_driver *drv, int value)
  804. {
  805. return (value == AMD7930_RATE) ? 0 : -EINVAL;
  806. }
  807. static int amd7930_get_output_muted(struct sparcaudio_driver *drv)
  808. {
  809.       return 0;
  810. }
  811. static void amd7930_loopback(struct sparcaudio_driver *drv, unsigned int value)
  812. {
  813. struct amd7930_info *info = (struct amd7930_info *) drv->private;
  814. if (value)
  815. info->map.mmr1 |= AM_MAP_MMR1_LOOPBACK;
  816. else
  817. info->map.mmr1 &= ~AM_MAP_MMR1_LOOPBACK;
  818. amd7930_update_map(drv);
  819. }
  820. static int amd7930_ioctl(struct inode * inode, struct file * file,
  821.                          unsigned int cmd, unsigned long arg, 
  822.                          struct sparcaudio_driver *drv)
  823. {
  824. int retval = 0;
  825.   
  826. switch (cmd) {
  827. case AUDIO_DIAG_LOOPBACK:
  828. amd7930_loopback(drv, (unsigned int)arg);
  829. break;
  830. default:
  831. retval = -EINVAL;
  832. };
  833. return retval;
  834. }
  835. /*
  836.  *       ISDN operations
  837.  *
  838.  * Many of these routines take an "int dev" argument, which is simply
  839.  * an index into the drivers[] array.  Currently, we only support a
  840.  * single AMD 7930 chip, so the value should always be 0.  B channel
  841.  * operations require an "int chan", which should be 0 for channel B1
  842.  * and 1 for channel B2
  843.  *
  844.  * int amd7930_get_irqnum(int dev)
  845.  *
  846.  *   returns the interrupt number being used by the chip.  ISDN4linux
  847.  *   uses this number to watch the interrupt during initialization and
  848.  *   make sure something is happening.
  849.  *
  850.  * int amd7930_get_liu_state(int dev)
  851.  *
  852.  *   returns the current state of the ISDN Line Interface Unit (LIU)
  853.  *   as a number between 2 (state F2) and 7 (state F7).  0 may also be
  854.  *   returned if the chip doesn't exist or the LIU hasn't been
  855.  *   activated.  The meanings of the states are defined in I.430, ISDN
  856.  *   BRI Physical Layer Interface.  The most important two states are
  857.  *   F3 (shutdown) and F7 (syncronized).
  858.  *
  859.  * void amd7930_liu_init(int dev, void (*callback)(), void *callback_arg)
  860.  *
  861.  *   initializes the LIU and optionally registers a callback to be
  862.  *   signaled upon a change of LIU state.  The callback will be called
  863.  *   with a single opaque callback_arg Once the callback has been
  864.  *   triggered, amd7930_get_liu_state can be used to determine the LIU
  865.  *   current state.
  866.  *
  867.  * void amd7930_liu_activate(int dev, int priority)
  868.  *
  869.  *   requests LIU activation at a given D-channel priority.
  870.  *   Successful activatation is achieved upon entering state F7, which
  871.  *   will trigger any callback previously registered with
  872.  *   amd7930_liu_init.
  873.  *
  874.  * void amd7930_liu_deactivate(int dev)
  875.  *
  876.  *   deactivates LIU.  Outstanding D and B channel transactions are
  877.  *   terminated rudely and without callback notification.  LIU change
  878.  *   of state callback will be triggered, however.
  879.  *
  880.  * void amd7930_dxmit(int dev, __u8 *buffer, unsigned int count,
  881.  *               void (*callback)(void *, int), void *callback_arg)
  882.  *
  883.  *   transmits a packet - specified with buffer, count - over the D-channel
  884.  *   interface.  Buffer should begin with the LAPD address field and
  885.  *   end with the information field.  FCS and flag sequences should not
  886.  *   be included, nor is bit-stuffing required - all these functions are
  887.  *   performed by the chip.  The callback function will be called
  888.  *   DURING THE TOP HALF OF AN INTERRUPT HANDLER and will be passed
  889.  *   both the arbitrary callback_arg and an integer error indication:
  890.  *
  891.  *       0 - successful transmission; ready for next packet
  892.  *   non-0 - error value from chip's DER (D-Channel Error Register):
  893.  *       4 - collision detect
  894.  *     128 - underrun; irq routine didn't service chip fast enough
  895.  *
  896.  *   The callback routine should defer any time-consuming operations
  897.  *   to a bottom-half handler; however, amd7930_dxmit may be called
  898.  *   from within the callback to request back-to-back transmission of
  899.  *   a second packet (without repeating the priority/collision mechanism)
  900.  *
  901.  *   A comment about the "collision detect" error, which is signalled
  902.  *   whenever the echoed D-channel data didn't match the transmitted
  903.  *   data.  This is part of ISDN's normal multi-drop T-interface
  904.  *   operation, indicating that another device has attempted simultaneous
  905.  *   transmission, but can also result from line noise.  An immediate
  906.  *   requeue via amd7930_dxmit is suggested, but repeated collision
  907.  *   errors may indicate a more serious problem.
  908.  *
  909.  * void amd7930_drecv(int dev, __u8 *buffer, unsigned int size,
  910.  *               void (*callback)(void *, int, unsigned int),
  911.  *               void *callback_arg)
  912.  *
  913.  *   register a buffer - buffer, size - into which a D-channel packet
  914.  *   can be received.  The callback function will be called DURING
  915.  *   THE TOP HALF OF AN INTERRUPT HANDLER and will be passed an
  916.  *   arbitrary callback_arg, an integer error indication and the length
  917.  *   of the received packet, which will start with the address field,
  918.  *   end with the information field, and not contain flag or FCS
  919.  *   bytes.  Bit-stuffing will already have been corrected for.
  920.  *   Possible values of second callback argument "error":
  921.  *
  922.  *       0 - successful reception
  923.  *   non-0 - error value from chip's DER (D-Channel Error Register):
  924.  *       1 - received packet abort
  925.  *       2 - framing error; non-integer number of bytes received
  926.  *       8 - FCS error; CRC sequence indicated corrupted data
  927.  *      16 - overflow error; packet exceeded size of buffer
  928.  *      32 - underflow error; packet smaller than required five bytes
  929.  *      64 - overrun error; irq routine didn't service chip fast enough
  930.  *
  931.  * int amd7930_bopen(int dev, int chan, u_char xmit_idle_char)
  932.  *
  933.  *   This function should be called before any other operations on a B
  934.  *   channel.  In addition to arranging for interrupt handling and
  935.  *   channel multiplexing, it sets the xmit_idle_char which is
  936.  *   transmitted on the interface when no data buffer is available.
  937.  *   Suggested values are: 0 for ISDN audio; FF for HDLC mark idle; 7E
  938.  *   for HDLC flag idle.  Returns 0 on a successful open; -1 on error,
  939.  *   which is quite possible if audio and the other ISDN channel are
  940.  *   already in use, since the Am7930 can only send two of the three
  941.  *   channels to the processor
  942.  *
  943.  * void amd7930_bclose(int dev, int chan)
  944.  *
  945.  *   Shuts down a B channel when no longer in use.
  946.  *
  947.  * void amd7930_bxmit(int dev, int chan, __u8 *buffer, unsigned int count,
  948.  *               void (*callback)(void *), void *callback_arg)
  949.  *
  950.  *   transmits a raw data block - specified with buffer, count - over
  951.  *   the B channel interface specified by dev/chan.  The callback
  952.  *   function will be called DURING THE TOP HALF OF AN INTERRUPT
  953.  *   HANDLER and will be passed the arbitrary callback_arg
  954.  *
  955.  *   The callback routine should defer any time-consuming operations
  956.  *   to a bottom-half handler; however, amd7930_bxmit may be called
  957.  *   from within the callback to request back-to-back transmission of
  958.  *   another data block
  959.  *
  960.  * void amd7930_brecv(int dev, int chan, __u8 *buffer, unsigned int size,
  961.  *               void (*callback)(void *), void *callback_arg)
  962.  *
  963.  *   receive a raw data block - specified with buffer, size - over the
  964.  *   B channel interface specified by dev/chan.  The callback function
  965.  *   will be called DURING THE TOP HALF OF AN INTERRUPT HANDLER and
  966.  *   will be passed the arbitrary callback_arg
  967.  *
  968.  *   The callback routine should defer any time-consuming operations
  969.  *   to a bottom-half handler; however, amd7930_brecv may be called
  970.  *   from within the callback to register another buffer and ensure
  971.  *   continuous B channel reception without loss of data
  972.  *
  973.  */
  974. #if defined (AMD79C30_ISDN)
  975. static int amd7930_get_irqnum(int dev)
  976. {
  977. struct amd7930_info *info;
  978. if (dev > num_drivers)
  979. return(0);
  980. info = (struct amd7930_info *) drivers[dev].private;
  981. return info->irq;
  982. }
  983. static int amd7930_get_liu_state(int dev)
  984. {
  985. struct amd7930_info *info;
  986. if (dev > num_drivers)
  987. return(0);
  988. info = (struct amd7930_info *) drivers[dev].private;
  989. return info->liu_state;
  990. }
  991. static void amd7930_liu_init(int dev, void (*callback)(), void *callback_arg)
  992. {
  993. struct amd7930_info *info;
  994. unsigned long flags;
  995. if (dev > num_drivers)
  996. return;
  997. info = (struct amd7930_info *) drivers[dev].private;
  998. save_and_cli(flags);
  999. /* Set callback for LIU state change */
  1000.         info->liu_callback = callback;
  1001. info->liu_callback_arg = callback_arg;
  1002. /* De-activate the ISDN Line Interface Unit (LIU) */
  1003. sbus_writeb(AMR_LIU_LMR1, info->regs + CR);
  1004. sbus_writeb(0, info->regs + DR);
  1005. /* Request interrupt when LIU changes state from/to F3/F7/F8 */
  1006. sbus_writeb(AMR_LIU_LMR2, info->regs + CR);
  1007. sbus_writeb(AM_LIU_LMR2_EN_F3_INT |
  1008.     AM_LIU_LMR2_EN_F7_INT |
  1009.     AM_LIU_LMR2_EN_F8_INT,
  1010.     info->regs + DR);
  1011. /* amd7930_enable_ints(info); */
  1012. /* Activate the ISDN Line Interface Unit (LIU) */
  1013. sbus_writeb(AMR_LIU_LMR1, info->regs + CR);
  1014. sbus_writeb(AM_LIU_LMR1_LIU_ENABL, info->regs + DR);
  1015. restore_flags(flags);
  1016. }
  1017. static void amd7930_liu_activate(int dev, int priority)
  1018. {
  1019. struct amd7930_info *info;
  1020. unsigned long flags;
  1021. if (dev > num_drivers)
  1022. return;
  1023. info = (struct amd7930_info *) drivers[dev].private;
  1024. save_and_cli(flags);
  1025.         /* Set D-channel access priority
  1026.          *
  1027.          * I.430 defines a priority mechanism based on counting 1s
  1028.          * in the echo channel before transmitting
  1029.          *
  1030.          * Priority 0 is eight 1s; priority 1 is ten 1s; etc
  1031.          */
  1032.         sbus_writeb(AMR_LIU_LPR, info->regs + CR);
  1033.         sbus_writeb(priority & 0x0f, info->regs + DR);
  1034. /* request LIU activation */
  1035. sbus_writeb(AMR_LIU_LMR1, info->regs + CR);
  1036. sbus_writeb(AM_LIU_LMR1_LIU_ENABL | AM_LIU_LMR1_REQ_ACTIV,
  1037.     info->regs + DR);
  1038. restore_flags(flags);
  1039. }
  1040. static void amd7930_liu_deactivate(int dev)
  1041. {
  1042. struct amd7930_info *info;
  1043. unsigned long flags;
  1044. if (dev > num_drivers)
  1045. return;
  1046. info = (struct amd7930_info *) drivers[dev].private;
  1047. save_and_cli(flags);
  1048. /* deactivate LIU */
  1049. sbus_writeb(AMR_LIU_LMR1, info->regs + CR);
  1050. sbus_writeb(0, info->regs + DR);
  1051. restore_flags(flags);
  1052. }
  1053. static void amd7930_dxmit(int dev, __u8 *buffer, unsigned int count,
  1054.   void (*callback)(void *, int), void *callback_arg)
  1055. {
  1056. struct amd7930_info *info;
  1057. unsigned long flags;
  1058. __u8 dmr1;
  1059. if (dev > num_drivers)
  1060. return;
  1061. info = (struct amd7930_info *) drivers[dev].private;
  1062. save_and_cli(flags);
  1063. if (info->D.output_ptr) {
  1064. restore_flags(flags);
  1065. printk("amd7930_dxmit: transmitter in usen");
  1066. return;
  1067. }
  1068. info->D.output_ptr = buffer;
  1069. info->D.output_count = count;
  1070. info->D.output_callback = callback;
  1071. info->D.output_callback_arg = callback_arg;
  1072. /* Enable D-channel Transmit Threshold interrupt; disable addressing */
  1073. sbus_writeb(AMR_DLC_DMR1, info->regs + CR);
  1074. dmr1 = sbus_readb(info->regs + DR);
  1075. dmr1 |= AMR_DLC_DMR1_DTTHRSH_INT;
  1076. dmr1 &= ~AMR_DLC_DMR1_EN_ADDRS;
  1077. sbus_writeb(dmr1, info->regs + DR);
  1078. /* Begin xmit by setting D-channel Transmit Byte Count Reg (DTCR) */
  1079. sbus_writeb(AMR_DLC_DTCR, info->regs + CR);
  1080. sbus_writeb(count & 0xff, info->regs + DR);
  1081. sbus_writeb((count >> 8) & 0xff, info->regs + DR);
  1082. /* Prime xmit FIFO */
  1083. /* fill_D_xmit_fifo(info); */
  1084. transceive_Dchannel(info);
  1085. restore_flags(flags);
  1086. }
  1087. static void amd7930_drecv(int dev, __u8 *buffer, unsigned int size,
  1088.   void (*callback)(void *, int, unsigned int),
  1089.   void *callback_arg)
  1090. {
  1091. struct amd7930_info *info;
  1092. unsigned long flags;
  1093. __u8 dmr1;
  1094. if (dev > num_drivers)
  1095. return;
  1096. info = (struct amd7930_info *) drivers[dev].private;
  1097. save_and_cli(flags);
  1098. if (info->D.input_ptr) {
  1099. restore_flags(flags);
  1100. printk("amd7930_drecv: receiver already has buffer!n");
  1101. return;
  1102. }
  1103. info->D.input_ptr = buffer;
  1104. info->D.input_count = 0;
  1105. info->D.input_limit = size;
  1106. info->D.input_callback = callback;
  1107. info->D.input_callback_arg = callback_arg;
  1108. /* Enable D-channel Receive Threshold interrupt;
  1109.  * Enable D-channel End of Receive Packet interrupt;
  1110.  * Disable address recognition
  1111.  */
  1112. sbus_writeb(AMR_DLC_DMR1, info->regs + CR);
  1113. dmr1 = sbus_readb(info->regs + DR);
  1114. dmr1 |= AMR_DLC_DMR1_DRTHRSH_INT | AMR_DLC_DMR1_EORP_INT;
  1115. dmr1 &= ~AMR_DLC_DMR1_EN_ADDRS;
  1116. sbus_writeb(dmr1, info->regs + DR);
  1117. /* Set D-channel Receive Byte Count Limit Register */
  1118. sbus_writeb(AMR_DLC_DRCR, info->regs + CR);
  1119. sbus_writeb(size & 0xff, info->regs + DR);
  1120. sbus_writeb((size >> 8) & 0xff, info->regs + DR);
  1121. restore_flags(flags);
  1122. }
  1123. static int amd7930_bopen(int dev, unsigned int chan, 
  1124.                          int mode, u_char xmit_idle_char)
  1125. {
  1126. struct amd7930_info *info;
  1127. unsigned long flags;
  1128. u8 tmp;
  1129. if (dev > num_drivers || chan<0 || chan>1)
  1130. return -1;
  1131. if (mode == L1_MODE_HDLC)
  1132. return -1;
  1133.  
  1134. info = (struct amd7930_info *) drivers[dev].private;
  1135. save_and_cli(flags);
  1136. if (info->Bb.channel_status == CHANNEL_AVAILABLE) {
  1137. info->Bb.channel_status = CHANNEL_INUSE;
  1138. info->Bb.xmit_idle_char = xmit_idle_char;
  1139. info->Bisdn[chan] = &info->Bb;
  1140. /* Multiplexor map - isdn (B1/2) to Bb */
  1141. sbus_writeb(AMR_MUX_MCR2 + chan, info->regs + CR);
  1142. sbus_writeb((AM_MUX_CHANNEL_B1 + chan) |
  1143.     (AM_MUX_CHANNEL_Bb << 4),
  1144.     info->regs + DR);
  1145. } else if (info->Bc.channel_status == CHANNEL_AVAILABLE) {
  1146. info->Bc.channel_status = CHANNEL_INUSE;
  1147. info->Bc.xmit_idle_char = xmit_idle_char;
  1148. info->Bisdn[chan] = &info->Bc;
  1149. /* Multiplexor map - isdn (B1/2) to Bc */
  1150. sbus_writeb(AMR_MUX_MCR2 + chan, info->regs + CR);
  1151. sbus_writeb((AM_MUX_CHANNEL_B1 + chan) |
  1152.     (AM_MUX_CHANNEL_Bc << 4),
  1153.     info->regs + DR);
  1154. } else {
  1155. restore_flags(flags);
  1156. return (-1);
  1157. }
  1158. /* Enable B channel transmit */
  1159. sbus_writeb(AMR_LIU_LMR1, info->regs + CR);
  1160. tmp = sbus_readb(info->regs + DR);
  1161. tmp |= AM_LIU_LMR1_B1_ENABL + chan;
  1162. sbus_writeb(tmp, info->regs + DR);
  1163. /* Enable B channel interrupts */
  1164. sbus_writeb(AMR_MUX_MCR4, info->regs + CR);
  1165. sbus_writeb(AM_MUX_MCR4_ENABLE_INTS |
  1166.     AM_MUX_MCR4_REVERSE_Bb |
  1167.     AM_MUX_MCR4_REVERSE_Bc,
  1168.     info->regs + DR);
  1169. restore_flags(flags);
  1170. return 0;
  1171. }
  1172. static void amd7930_bclose(int dev, unsigned int chan)
  1173. {
  1174. struct amd7930_info *info;
  1175. unsigned long flags;
  1176. if (dev > num_drivers || chan<0 || chan>1)
  1177. return;
  1178. info = (struct amd7930_info *) drivers[dev].private;
  1179. save_and_cli(flags);
  1180. if (info->Bisdn[chan]) {
  1181. u8 tmp;
  1182. info->Bisdn[chan]->channel_status = CHANNEL_AVAILABLE;
  1183. sbus_writeb(AMR_MUX_MCR2 + chan, info->regs + CR);
  1184. sbus_writeb(0, info->regs + DR);
  1185. info->Bisdn[chan] = NULL;
  1186. /* Disable B channel transmit */
  1187. sbus_writeb(AMR_LIU_LMR1, info->regs + CR);
  1188. tmp = sbus_readb(info->regs + DR);
  1189. tmp &= ~(AM_LIU_LMR1_B1_ENABL + chan);
  1190. sbus_writeb(tmp, info->regs + DR);
  1191. if (info->Bb.channel_status == CHANNEL_AVAILABLE &&
  1192.     info->Bc.channel_status == CHANNEL_AVAILABLE) {
  1193. /* Disable B channel interrupts */
  1194. sbus_writeb(AMR_MUX_MCR4, info->regs + CR);
  1195. sbus_writeb(0, info->regs + DR);
  1196. }
  1197. }
  1198. restore_flags(flags);
  1199. }
  1200. static void amd7930_bxmit(int dev, unsigned int chan,
  1201.                           __u8 * buffer, unsigned long count,
  1202.   void (*callback)(void *, int), void *callback_arg)
  1203. {
  1204. struct amd7930_info *info;
  1205. struct amd7930_channel *Bchan;
  1206. unsigned long flags;
  1207. if (dev > num_drivers)
  1208. return;
  1209. info = (struct amd7930_info *) drivers[dev].private;
  1210. Bchan = info->Bisdn[chan];
  1211. if (Bchan) {
  1212. save_and_cli(flags);
  1213. Bchan->output_ptr = buffer;
  1214. Bchan->output_count = count;
  1215. Bchan->output_format = AUDIO_ENCODING_ULAW;
  1216.         Bchan->output_callback = (void *) callback;
  1217.          Bchan->output_callback_arg = callback_arg;
  1218. restore_flags(flags);
  1219. }
  1220. }
  1221. static void amd7930_brecv(int dev, unsigned int chan, 
  1222.                           __u8 * buffer, unsigned long size,
  1223.                           void (*callback)(void *, int, unsigned int),
  1224.                           void *callback_arg)
  1225. {
  1226. struct amd7930_info *info;
  1227. struct amd7930_channel *Bchan;
  1228. unsigned long flags;
  1229. if (dev > num_drivers)
  1230. return;
  1231. info = (struct amd7930_info *) drivers[dev].private;
  1232. Bchan = info->Bisdn[chan];
  1233. if (Bchan) {
  1234. save_and_cli(flags);
  1235. Bchan->input_ptr = buffer;
  1236. Bchan->input_count = size;
  1237. Bchan->input_format = AUDIO_ENCODING_ULAW;
  1238. Bchan->input_callback = (void *) callback;
  1239. Bchan->input_callback_arg = callback_arg;
  1240. restore_flags(flags);
  1241. }
  1242. }
  1243. struct foreign_interface amd7930_foreign_interface = {
  1244.         amd7930_get_irqnum,
  1245.         amd7930_get_liu_state,
  1246.         amd7930_liu_init,
  1247.         amd7930_liu_activate,
  1248.         amd7930_liu_deactivate,
  1249.         amd7930_dxmit,
  1250.         amd7930_drecv,
  1251.         amd7930_bopen,
  1252.         amd7930_bclose,
  1253.         amd7930_bxmit,
  1254.         amd7930_brecv
  1255. };
  1256. EXPORT_SYMBOL(amd7930_foreign_interface);
  1257. #endif
  1258. /*
  1259.  * Device detection and initialization.
  1260.  */
  1261. static struct sparcaudio_operations amd7930_ops = {
  1262. amd7930_open,
  1263. amd7930_release,
  1264. amd7930_ioctl,
  1265. amd7930_start_output,
  1266. amd7930_stop_output,
  1267. amd7930_start_input,
  1268. amd7930_stop_input,
  1269. amd7930_sunaudio_getdev,
  1270. amd7930_set_output_volume,
  1271. amd7930_get_output_volume,
  1272. amd7930_set_input_volume,
  1273. amd7930_get_input_volume,
  1274. amd7930_set_monitor_volume,
  1275. amd7930_get_monitor_volume,
  1276. NULL, /* amd7930_set_output_balance */
  1277. amd7930_get_output_balance,
  1278. NULL, /* amd7930_set_input_balance */
  1279. amd7930_get_input_balance,
  1280. amd7930_set_output_channels,
  1281. amd7930_get_output_channels,
  1282. amd7930_set_input_channels,
  1283. amd7930_get_input_channels,
  1284. amd7930_set_output_precision,
  1285. amd7930_get_output_precision,
  1286. amd7930_set_input_precision,
  1287. amd7930_get_input_precision,
  1288. amd7930_set_output_port,
  1289. amd7930_get_output_port,
  1290. NULL, /* amd7930_set_input_port */
  1291. amd7930_get_input_port,
  1292. amd7930_set_encoding,
  1293. amd7930_get_encoding,
  1294. amd7930_set_encoding,
  1295. amd7930_get_encoding,
  1296. amd7930_set_output_rate,
  1297. amd7930_get_output_rate,
  1298. amd7930_set_input_rate,
  1299. amd7930_get_input_rate,
  1300. amd7930_sunaudio_getdev_sunos,
  1301. amd7930_get_output_ports,
  1302. amd7930_get_input_ports,
  1303. NULL,                    /* amd7930_set_output_muted */
  1304. amd7930_get_output_muted,
  1305.         NULL,                   /* amd7930_set_output_pause */
  1306.         NULL,                   /* amd7930_get_output_pause */
  1307.         NULL,                   /* amd7930_set_input_pause */
  1308.         NULL,                   /* amd7930_get_input_pause */
  1309.         NULL,                   /* amd7930_set_output_samples */
  1310.         NULL,                   /* amd7930_get_output_samples */
  1311.         NULL,                   /* amd7930_set_input_samples */
  1312.         NULL,                   /* amd7930_get_input_samples */
  1313.         NULL,                   /* amd7930_set_output_error */
  1314.         NULL,                   /* amd7930_get_output_error */
  1315.         NULL,                   /* amd7930_set_input_error */
  1316.         NULL,                   /* amd7930_get_input_error */
  1317.         amd7930_get_formats,
  1318. };
  1319. /* Attach to an amd7930 chip given its PROM node. */
  1320. static int amd7930_attach(struct sparcaudio_driver *drv, int node,
  1321.   struct sbus_bus *sbus, struct sbus_dev *sdev)
  1322. {
  1323. struct linux_prom_registers regs;
  1324. struct linux_prom_irqs irq;
  1325. struct resource res, *resp;
  1326. struct amd7930_info *info;
  1327. int err;
  1328. /* Allocate our private information structure. */
  1329. drv->private = kmalloc(sizeof(struct amd7930_info), GFP_KERNEL);
  1330. if (drv->private == NULL)
  1331. return -ENOMEM;
  1332. /* Point at the information structure and initialize it. */
  1333. drv->ops = &amd7930_ops;
  1334. info = (struct amd7930_info *)drv->private;
  1335. memset(info, 0, sizeof(*info));
  1336. info->ints_on = 1; /* force disable below */
  1337. drv->dev = sdev;
  1338. /* Map the registers into memory. */
  1339. prom_getproperty(node, "reg", (char *)&regs, sizeof(regs));
  1340. if (sbus && sdev) {
  1341. resp = &sdev->resource[0];
  1342. } else {
  1343. resp = &res;
  1344. res.start = regs.phys_addr;
  1345. res.end = res.start + regs.reg_size - 1;
  1346. res.flags = IORESOURCE_IO | (regs.which_io & 0xff);
  1347. }
  1348. info->regs_size = regs.reg_size;
  1349. info->regs = sbus_ioremap(resp, 0, regs.reg_size, "amd7930");
  1350. if (!info->regs) {
  1351. printk(KERN_ERR "amd7930: could not remap registersn");
  1352. kfree(drv->private);
  1353. return -EIO;
  1354. }
  1355. /* Put amd7930 in idle mode (interrupts disabled) */
  1356. amd7930_idle(info);
  1357. /* Enable extended FIFO operation on D-channel */
  1358. sbus_writeb(AMR_DLC_EFCR, info->regs + CR);
  1359. sbus_writeb(AMR_DLC_EFCR_EXTEND_FIFO, info->regs + DR);
  1360. sbus_writeb(AMR_DLC_DMR4, info->regs + CR);
  1361. sbus_writeb(/* AMR_DLC_DMR4_RCV_30 | */ AMR_DLC_DMR4_XMT_14,
  1362.     info->regs + DR);
  1363. /* Attach the interrupt handler to the audio interrupt. */
  1364. prom_getproperty(node, "intr", (char *)&irq, sizeof(irq));
  1365. info->irq = irq.pri;
  1366. request_irq(info->irq, amd7930_interrupt,
  1367.     SA_INTERRUPT, "amd7930", drv);
  1368. enable_irq(info->irq);
  1369. amd7930_enable_ints(info);
  1370. /* Initalize the local copy of the MAP registers. */
  1371. memset(&info->map, 0, sizeof(info->map));
  1372. info->map.mmr1 = AM_MAP_MMR1_GX | AM_MAP_MMR1_GER |
  1373.  AM_MAP_MMR1_GR | AM_MAP_MMR1_STG;
  1374.         /* Start out with speaker, microphone */
  1375.         info->map.mmr2 |= (AM_MAP_MMR2_LS | AM_MAP_MMR2_AINB);
  1376. /* Set the default audio parameters. */
  1377. info->rgain = 128;
  1378. info->pgain = 200;
  1379. info->mgain = 0;
  1380. info->format_type = AUDIO_ENCODING_ULAW;
  1381. info->Bb.input_format = AUDIO_ENCODING_ULAW;
  1382. info->Bb.output_format = AUDIO_ENCODING_ULAW;
  1383. info->Bc.input_format = AUDIO_ENCODING_ULAW;
  1384. info->Bc.output_format = AUDIO_ENCODING_ULAW;
  1385. amd7930_update_map(drv);
  1386. /* Register the amd7930 with the midlevel audio driver. */
  1387. err = register_sparcaudio_driver(drv, 1);
  1388. if (err < 0) {
  1389. printk(KERN_ERR "amd7930: unable to registern");
  1390. disable_irq(info->irq);
  1391. free_irq(info->irq, drv);
  1392. sbus_iounmap(info->regs, info->regs_size);
  1393. kfree(drv->private);
  1394. return -EIO;
  1395. }
  1396. /* Announce the hardware to the user. */
  1397. printk(KERN_INFO "amd7930 at %lx irq %dn",
  1398.        info->regs, info->irq);
  1399. /* Success! */
  1400. return 0;
  1401. }
  1402. /* Detach from an amd7930 chip given the device structure. */
  1403. static void __exit amd7930_detach(struct sparcaudio_driver *drv)
  1404. {
  1405. struct amd7930_info *info = (struct amd7930_info *)drv->private;
  1406. unregister_sparcaudio_driver(drv, 1);
  1407. amd7930_idle(info);
  1408. disable_irq(info->irq);
  1409. free_irq(info->irq, drv);
  1410. sbus_iounmap(info->regs, info->regs_size);
  1411. kfree(drv->private);
  1412. }
  1413. /* Probe for the amd7930 chip and then attach the driver. */
  1414. static int __init amd7930_init(void)
  1415. {
  1416. struct sbus_bus *sbus;
  1417. struct sbus_dev *sdev;
  1418. int node;
  1419. /* Try to find the sun4c "audio" node first. */
  1420. node = prom_getchild(prom_root_node);
  1421. node = prom_searchsiblings(node, "audio");
  1422. if (node && amd7930_attach(&drivers[0], node, NULL, NULL) == 0)
  1423. num_drivers = 1;
  1424. else
  1425. num_drivers = 0;
  1426. /* Probe each SBUS for amd7930 chips. */
  1427. for_all_sbusdev(sdev, sbus) {
  1428. if (!strcmp(sdev->prom_name, "audio")) {
  1429. /* Don't go over the max number of drivers. */
  1430. if (num_drivers >= MAX_DRIVERS)
  1431. continue;
  1432. if (amd7930_attach(&drivers[num_drivers],
  1433.    sdev->prom_node, sdev->bus, sdev) == 0)
  1434. num_drivers++;
  1435. }
  1436. }
  1437. /* Only return success if we found some amd7930 chips. */
  1438. return (num_drivers > 0) ? 0 : -EIO;
  1439. }
  1440. static void __exit amd7930_exit(void)
  1441. {
  1442. register int i;
  1443. for (i = 0; i < num_drivers; i++) {
  1444. amd7930_detach(&drivers[i]);
  1445. num_drivers--;
  1446. }
  1447. }
  1448. module_init(amd7930_init);
  1449. module_exit(amd7930_exit);
  1450. MODULE_LICENSE("GPL");
  1451. /*************************************************************/
  1452. /*                 Audio format conversion                   */
  1453. /*************************************************************/
  1454. /* Translation tables */
  1455. static unsigned char ulaw[] = {
  1456.     3,   7,  11,  15,  19,  23,  27,  31, 
  1457.    35,  39,  43,  47,  51,  55,  59,  63, 
  1458.    66,  68,  70,  72,  74,  76,  78,  80, 
  1459.    82,  84,  86,  88,  90,  92,  94,  96, 
  1460.    98,  99, 100, 101, 102, 103, 104, 105, 
  1461.   106, 107, 108, 109, 110, 111, 112, 113, 
  1462.   113, 114, 114, 115, 115, 116, 116, 117, 
  1463.   117, 118, 118, 119, 119, 120, 120, 121, 
  1464.   121, 121, 122, 122, 122, 122, 123, 123, 
  1465.   123, 123, 124, 124, 124, 124, 125, 125, 
  1466.   125, 125, 125, 125, 126, 126, 126, 126, 
  1467.   126, 126, 126, 126, 127, 127, 127, 127, 
  1468.   127, 127, 127, 127, 127, 127, 127, 127, 
  1469.   128, 128, 128, 128, 128, 128, 128, 128, 
  1470.   128, 128, 128, 128, 128, 128, 128, 128, 
  1471.   128, 128, 128, 128, 128, 128, 128, 128, 
  1472.   253, 249, 245, 241, 237, 233, 229, 225, 
  1473.   221, 217, 213, 209, 205, 201, 197, 193, 
  1474.   190, 188, 186, 184, 182, 180, 178, 176, 
  1475.   174, 172, 170, 168, 166, 164, 162, 160, 
  1476.   158, 157, 156, 155, 154, 153, 152, 151, 
  1477.   150, 149, 148, 147, 146, 145, 144, 143, 
  1478.   143, 142, 142, 141, 141, 140, 140, 139, 
  1479.   139, 138, 138, 137, 137, 136, 136, 135, 
  1480.   135, 135, 134, 134, 134, 134, 133, 133, 
  1481.   133, 133, 132, 132, 132, 132, 131, 131, 
  1482.   131, 131, 131, 131, 130, 130, 130, 130, 
  1483.   130, 130, 130, 130, 129, 129, 129, 129, 
  1484.   129, 129, 129, 129, 129, 129, 129, 129, 
  1485.   128, 128, 128, 128, 128, 128, 128, 128, 
  1486.   128, 128, 128, 128, 128, 128, 128, 128, 
  1487.   128, 128, 128, 128, 128, 128, 128, 128
  1488. };
  1489. static __u8 mulaw2bilinear(__u8 data)
  1490. {
  1491. return ulaw[data];
  1492. }
  1493. static unsigned char linear[] = {
  1494.      0,    0,    0,    0,    0,    0,    0,    1,
  1495.      0,    0,    0,    2,    0,    0,    0,    3,
  1496.      0,    0,    0,    4,    0,    0,    0,    5,
  1497.      0,    0,    0,    6,    0,    0,    0,    7,
  1498.      0,    0,    0,    8,    0,    0,    0,    9,
  1499.      0,    0,    0,   10,    0,    0,    0,   11,
  1500.      0,    0,    0,   12,    0,    0,    0,   13,
  1501.      0,    0,    0,   14,    0,    0,    0,   15,
  1502.      0,    0,   16,    0,   17,    0,   18,    0,
  1503.     19,    0,   20,    0,   21,    0,   22,    0,
  1504.     23,    0,   24,    0,   25,    0,   26,    0,
  1505.     27,    0,   28,    0,   29,    0,   30,    0,
  1506.     31,    0,   32,   33,   34,   35,   36,   37,
  1507.     38,   39,   40,   41,   42,   43,   44,   45,
  1508.     46,   48,   50,   52,   54,   56,   58,   60,
  1509.     62,   65,   69,   73,   77,   83,   91,  103,
  1510.    255,  231,  219,  211,  205,  201,  197,  193,
  1511.    190,  188,  186,  184,  182,  180,  178,  176,
  1512.    174,  173,  172,  171,  170,  169,  168,  167,
  1513.    166,  165,  164,  163,  162,  161,  160,    0,
  1514.    159,    0,  158,    0,  157,    0,  156,    0,
  1515.    155,    0,  154,    0,  153,    0,  152,    0,
  1516.    151,    0,  150,    0,  149,    0,  148,    0,
  1517.    147,    0,  146,    0,  145,    0,  144,    0,
  1518.      0,  143,    0,    0,    0,  142,    0,    0,
  1519.      0,  141,    0,    0,    0,  140,    0,    0,
  1520.      0,  139,    0,    0,    0,  138,    0,    0,
  1521.      0,  137,    0,    0,    0,  136,    0,    0,
  1522.      0,  135,    0,    0,    0,  134,    0,    0,
  1523.      0,  133,    0,    0,    0,  132,    0,    0,
  1524.      0,  131,    0,    0,    0,  130,    0,    0,
  1525.      0,  129,    0,    0,    0,  128,    0,    0
  1526. };
  1527. static __u8 bilinear2mulaw(__u8 data)
  1528. {
  1529. return linear[data];
  1530. }
  1531. static int exp_lut[256] = {
  1532. 0,0,1,1,2,2,2,2,3,3,3,3,3,3,3,3,4,4,4,4,4,4,4,4,4,4,4,4,4,4,4,4,
  1533. 5,5,5,5,5,5,5,5,5,5,5,5,5,5,5,5,5,5,5,5,5,5,5,5,5,5,5,5,5,5,5,5,
  1534. 6,6,6,6,6,6,6,6,6,6,6,6,6,6,6,6,6,6,6,6,6,6,6,6,6,6,6,6,6,6,6,6,
  1535. 6,6,6,6,6,6,6,6,6,6,6,6,6,6,6,6,6,6,6,6,6,6,6,6,6,6,6,6,6,6,6,6,
  1536. 7,7,7,7,7,7,7,7,7,7,7,7,7,7,7,7,7,7,7,7,7,7,7,7,7,7,7,7,7,7,7,7,
  1537. 7,7,7,7,7,7,7,7,7,7,7,7,7,7,7,7,7,7,7,7,7,7,7,7,7,7,7,7,7,7,7,7,
  1538. 7,7,7,7,7,7,7,7,7,7,7,7,7,7,7,7,7,7,7,7,7,7,7,7,7,7,7,7,7,7,7,7,
  1539. 7,7,7,7,7,7,7,7,7,7,7,7,7,7,7,7,7,7,7,7,7,7,7,7,7,7,7,7,7,7,7,7
  1540. };
  1541. #define BIAS 0x84
  1542. #define CLIP 32635
  1543. #define SWAP_ENDIAN(x) ((x >> 8) | ((x & 0xff) << 8))
  1544. static __u8  linear2mulaw(__u16 data)
  1545. {
  1546. static int sign, exponent, mantissa;
  1547. /* not really sure, if swapping is ok - comment next line to disable it */
  1548. data = SWAP_ENDIAN(data);
  1549. sign = (data >> 8) & 0x80;
  1550. if (sign != 0) data = -data;
  1551. if (data > CLIP) data = CLIP;
  1552. data += BIAS;
  1553. exponent = exp_lut[(data >> 7) & 0xFF];
  1554. mantissa = (data >> (exponent + 3)) & 0x0F;
  1555. return (~(sign | (exponent << 4) | mantissa));
  1556. }
  1557. static __u16 mulaw2linear(__u8 data)
  1558. {
  1559. /* this conversion is not working */
  1560. return data;
  1561. }
  1562. #if 0
  1563. #define INOUT(x,y) (((x) << 16) | (y))
  1564. static int convert_audio(int in_format, int out_format, __u8* buffer, int count)
  1565. {
  1566. static int i,sign,exponent;
  1567. static __u16 data;
  1568. if (in_format == out_format) return count;
  1569. switch(INOUT(in_format, out_format)) {
  1570. case INOUT(AUDIO_ENCODING_ULAW, AUDIO_ENCODING_LINEAR8):
  1571. for (i = 0;i < count; i++) {
  1572. buffer[i] = ulaw[buffer[i]];
  1573. };
  1574. break;
  1575. case INOUT(AUDIO_ENCODING_ULAW, AUDIO_ENCODING_LINEAR):
  1576. break;
  1577. case INOUT(AUDIO_ENCODING_LINEAR, AUDIO_ENCODING_ULAW):
  1578. /* buffer is two-byte => convert to first */
  1579. for (i = 0; i < count/2; i++) {
  1580. data = ((__u16*)buffer)[i];
  1581. sign = (data >> 8) & 0x80;
  1582. if (data > CLIP) data = CLIP;
  1583. data += BIAS;
  1584. exponent = exp_lut[(data >> 7) & 0xFF];
  1585. buffer[i] = ~(sign | (exponent << 4) | 
  1586.   ((data >> (exponent + 3)) & 0x0F));
  1587. };
  1588. break;
  1589. case INOUT(AUDIO_ENCODING_LINEAR8, AUDIO_ENCODING_ULAW):
  1590. for (i = 0; i < count; i++) {
  1591. buffer[i] = linear[buffer[i]];
  1592. };
  1593. break;
  1594. default:
  1595. return 0;
  1596. };
  1597. return count;
  1598. }
  1599. #undef INOUT
  1600. #endif
  1601. #undef BIAS
  1602. #undef CLIP
  1603. #undef SWAP_ENDIAN