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

Linux/Unix编程

开发平台:

Unix_Linux

  1. /*
  2.  * HP i8042-based System Device Controller driver.
  3.  *
  4.  * Copyright (c) 2001 Brian S. Julin
  5.  * All rights reserved.
  6.  *
  7.  * Redistribution and use in source and binary forms, with or without
  8.  * modification, are permitted provided that the following conditions
  9.  * are met:
  10.  * 1. Redistributions of source code must retain the above copyright
  11.  *    notice, this list of conditions, and the following disclaimer,
  12.  *    without modification.
  13.  * 2. The name of the author may not be used to endorse or promote products
  14.  *    derived from this software without specific prior written permission.
  15.  *
  16.  * Alternatively, this software may be distributed under the terms of the
  17.  * GNU General Public License ("GPL").
  18.  *
  19.  * THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND
  20.  * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
  21.  * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
  22.  * ARE DISCLAIMED. IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE FOR
  23.  * ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
  24.  * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
  25.  * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
  26.  * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
  27.  * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
  28.  *
  29.  * References:
  30.  * System Device Controller Microprocessor Firmware Theory of Operation
  31.  *      for Part Number 1820-4784 Revision B.  Dwg No. A-1820-4784-2
  32.  * Helge Deller's original hilkbd.c port for PA-RISC.
  33.  *
  34.  *
  35.  * Driver theory of operation:
  36.  *
  37.  * hp_sdc_put does all writing to the SDC.  ISR can run on a different 
  38.  * CPU than hp_sdc_put, but only one CPU runs hp_sdc_put at a time 
  39.  * (it cannot really benefit from SMP anyway.)  A tasket fit this perfectly.
  40.  *
  41.  * All data coming back from the SDC is sent via interrupt and can be read 
  42.  * fully in the ISR, so there are no latency/throughput problems there.  
  43.  * The problem is with output, due to the slow clock speed of the SDC 
  44.  * compared to the CPU.  This should not be too horrible most of the time, 
  45.  * but if used with HIL devices that support the multibyte transfer command, 
  46.  * keeping outbound throughput flowing at the 6500KBps that the HIL is 
  47.  * capable of is more than can be done at HZ=100.
  48.  *
  49.  * Busy polling for IBF clear wastes CPU cycles and bus cycles.  hp_sdc.ibf 
  50.  * is set to 0 when the IBF flag in the status register has cleared.  ISR 
  51.  * may do this, and may also access the parts of queued transactions related 
  52.  * to reading data back from the SDC, but otherwise will not touch the 
  53.  * hp_sdc state. Whenever a register is written hp_sdc.ibf is set to 1.
  54.  *
  55.  * The i8042 write index and the values in the 4-byte input buffer
  56.  * starting at 0x70 are kept track of in hp_sdc.wi, and .r7[], respectively,
  57.  * to minimize the amount of IO needed to the SDC.  However these values 
  58.  * do not need to be locked since they are only ever accessed by hp_sdc_put.
  59.  *
  60.  * A timer task schedules the tasklet once per second just to make
  61.  * sure it doesn't freeze up and to allow for bad reads to time out.
  62.  */
  63. #include <linux/hp_sdc.h>
  64. #include <linux/sched.h>
  65. #include <linux/errno.h>
  66. #include <linux/init.h>
  67. #include <linux/module.h>
  68. #include <linux/ioport.h>
  69. #include <linux/time.h>
  70. #include <linux/slab.h>
  71. #include <linux/hil.h>
  72. #include <asm/io.h>
  73. #include <asm/system.h>
  74. #include <asm/gsc.h>
  75. #define PREFIX "HP SDC: "
  76. MODULE_AUTHOR("Brian S. Julin <bri@calyx.com>");
  77. MODULE_DESCRIPTION("HP i8042-based SDC Driver");
  78. MODULE_LICENSE("Dual BSD/GPL");
  79. EXPORT_SYMBOL(hp_sdc_request_timer_irq);
  80. EXPORT_SYMBOL(hp_sdc_request_hil_irq);
  81. EXPORT_SYMBOL(hp_sdc_request_cooked_irq);
  82. EXPORT_SYMBOL(hp_sdc_release_timer_irq);
  83. EXPORT_SYMBOL(hp_sdc_release_hil_irq);
  84. EXPORT_SYMBOL(hp_sdc_release_cooked_irq);
  85. EXPORT_SYMBOL(hp_sdc_enqueue_transaction);
  86. EXPORT_SYMBOL(hp_sdc_dequeue_transaction);
  87. static hp_i8042_sdc hp_sdc; /* All driver state is kept in here. */
  88. /*************** primitives for use in any context *********************/
  89. static inline uint8_t hp_sdc_status_in8 (void) {
  90. uint8_t status;
  91. unsigned long flags;
  92. write_lock_irqsave(&hp_sdc.ibf_lock, flags);
  93. status = gsc_readb(hp_sdc.status_io);
  94. if (!(status & HP_SDC_STATUS_IBF)) hp_sdc.ibf = 0;
  95. write_unlock_irqrestore(&hp_sdc.ibf_lock, flags);
  96. return status;
  97. }
  98. static inline uint8_t hp_sdc_data_in8 (void) {
  99. return gsc_readb(hp_sdc.data_io); 
  100. }
  101. static inline void hp_sdc_status_out8 (uint8_t val) {
  102. unsigned long flags;
  103. write_lock_irqsave(&hp_sdc.ibf_lock, flags);
  104. hp_sdc.ibf = 1;
  105. if ((val & 0xf0) == 0xe0) hp_sdc.wi = 0xff;
  106. gsc_writeb(val, hp_sdc.status_io);
  107. write_unlock_irqrestore(&hp_sdc.ibf_lock, flags);
  108. }
  109. static inline void hp_sdc_data_out8 (uint8_t val) {
  110. unsigned long flags;
  111. write_lock_irqsave(&hp_sdc.ibf_lock, flags);
  112. hp_sdc.ibf = 1;
  113. gsc_writeb(val, hp_sdc.data_io);
  114. write_unlock_irqrestore(&hp_sdc.ibf_lock, flags);
  115. }
  116. /* Care must be taken to only invoke hp_sdc_spin_ibf when 
  117.  * absolutely needed, or in rarely invoked subroutines.  
  118.  * Not only does it waste CPU cycles, it also wastes bus cycles. 
  119.  */
  120. static inline void hp_sdc_spin_ibf(void) {
  121. unsigned long flags;
  122. rwlock_t *lock;
  123. lock = &hp_sdc.ibf_lock;
  124. read_lock_irqsave(lock, flags);
  125. if (!hp_sdc.ibf) {
  126. read_unlock_irqrestore(lock, flags);
  127. return;
  128. }
  129. read_unlock(lock);
  130. write_lock(lock);
  131. while (gsc_readb(hp_sdc.status_io) & HP_SDC_STATUS_IBF) {};
  132. hp_sdc.ibf = 0;
  133. write_unlock_irqrestore(lock, flags);
  134. }
  135. /************************ Interrupt context functions ************************/
  136. static void hp_sdc_take (int irq, void *dev_id, uint8_t status, uint8_t data) {
  137. hp_sdc_transaction *curr;
  138. read_lock(&hp_sdc.rtq_lock);
  139. if (hp_sdc.rcurr < 0) {
  140.    read_unlock(&hp_sdc.rtq_lock);
  141. return;
  142. }
  143. curr = hp_sdc.tq[hp_sdc.rcurr];
  144. read_unlock(&hp_sdc.rtq_lock);
  145. curr->seq[curr->idx++] = status;
  146. curr->seq[curr->idx++] = data;
  147. hp_sdc.rqty -= 2;
  148. do_gettimeofday(&hp_sdc.rtv);
  149. if (hp_sdc.rqty <= 0) {
  150. /* All data has been gathered. */
  151. if(curr->seq[curr->actidx] & HP_SDC_ACT_SEMAPHORE) {
  152. if (curr->act.semaphore) up(curr->act.semaphore);
  153. }
  154. if(curr->seq[curr->actidx] & HP_SDC_ACT_CALLBACK) {
  155. if (curr->act.irqhook)
  156. curr->act.irqhook(irq, dev_id, status, data);
  157. }
  158. curr->actidx = curr->idx;
  159. curr->idx++;
  160. /* Return control of this transaction */
  161. write_lock(&hp_sdc.rtq_lock);
  162. hp_sdc.rcurr = -1; 
  163. hp_sdc.rqty = 0;
  164. write_unlock(&hp_sdc.rtq_lock);
  165. tasklet_schedule(&hp_sdc.task);
  166. }
  167. }
  168. static void hp_sdc_isr(int irq, void *dev_id, struct pt_regs * regs) {
  169. uint8_t status, data;
  170. status = hp_sdc_status_in8();
  171. /* Read data unconditionally to advance i8042. */
  172. data =   hp_sdc_data_in8();
  173. /* For now we are ignoring these until we get the SDC to behave. */
  174. if (((status & 0xf1) == 0x51) && data == 0x82) {
  175.   return;
  176. }
  177. switch(status & HP_SDC_STATUS_IRQMASK) {
  178.       case 0: /* This case is not documented. */
  179. break;
  180.       case HP_SDC_STATUS_USERTIMER:
  181.       case HP_SDC_STATUS_PERIODIC:
  182.       case HP_SDC_STATUS_TIMER:
  183. read_lock(&hp_sdc.hook_lock);
  184.        if (hp_sdc.timer != NULL)
  185. hp_sdc.timer(irq, dev_id, status, data);
  186. read_unlock(&hp_sdc.hook_lock);
  187. break;
  188.       case HP_SDC_STATUS_REG:
  189. hp_sdc_take(irq, dev_id, status, data);
  190. break;
  191.       case HP_SDC_STATUS_HILCMD:
  192.       case HP_SDC_STATUS_HILDATA:
  193. read_lock(&hp_sdc.hook_lock);
  194. if (hp_sdc.hil != NULL)
  195. hp_sdc.hil(irq, dev_id, status, data);
  196. read_unlock(&hp_sdc.hook_lock);
  197. break;
  198.       case HP_SDC_STATUS_PUP:
  199. read_lock(&hp_sdc.hook_lock);
  200. if (hp_sdc.pup != NULL)
  201. hp_sdc.pup(irq, dev_id, status, data);
  202. else printk(KERN_INFO PREFIX "HP SDC reports successful PUP.n");
  203. read_unlock(&hp_sdc.hook_lock);
  204. break;
  205.       default:
  206. read_lock(&hp_sdc.hook_lock);
  207. if (hp_sdc.cooked != NULL)
  208. hp_sdc.cooked(irq, dev_id, status, data);
  209. read_unlock(&hp_sdc.hook_lock);
  210. break;
  211. }
  212. }
  213. static void hp_sdc_nmisr(int irq, void *dev_id, struct pt_regs * regs) {
  214. int status;
  215. status = hp_sdc_status_in8();
  216. printk(KERN_WARNING PREFIX "NMI !n");
  217. #if 0
  218. if (status & HP_SDC_NMISTATUS_FHS) {
  219. read_lock(&hp_sdc.hook_lock);
  220.        if (hp_sdc.timer != NULL)
  221. hp_sdc.timer(irq, dev_id, status, 0);
  222. read_unlock(&hp_sdc.hook_lock);
  223. }
  224. else {
  225. /* TODO: pass this on to the HIL handler, or do SAK here? */
  226. printk(KERN_WARNING PREFIX "HIL NMIn");
  227. }
  228. #endif
  229. }
  230. /***************** Kernel (tasklet) context functions ****************/
  231. unsigned long hp_sdc_put(void);
  232. static void hp_sdc_tasklet(unsigned long foo) {
  233. write_lock_irq(&hp_sdc.rtq_lock);
  234. if (hp_sdc.rcurr >= 0) {
  235. struct timeval tv;
  236. do_gettimeofday(&tv);
  237. if (tv.tv_sec > hp_sdc.rtv.tv_sec) tv.tv_usec += 1000000;
  238. if (tv.tv_usec - hp_sdc.rtv.tv_usec > HP_SDC_MAX_REG_DELAY) {
  239. hp_sdc_transaction *curr;
  240. uint8_t tmp;
  241. curr = hp_sdc.tq[hp_sdc.rcurr];
  242. /* If this turns out to be a normal failure mode
  243.  * we'll need to figure out a way to communicate
  244.  * it back to the application. and be less verbose.
  245.  */
  246. printk(KERN_WARNING PREFIX "read timeout (%ius)!n",
  247.        tv.tv_usec - hp_sdc.rtv.tv_usec);
  248. curr->idx += hp_sdc.rqty;
  249. hp_sdc.rqty = 0;
  250. tmp = curr->seq[curr->actidx];
  251. curr->seq[curr->actidx] |= HP_SDC_ACT_DEAD;
  252. if(tmp & HP_SDC_ACT_SEMAPHORE) {
  253. if (curr->act.semaphore) 
  254. up(curr->act.semaphore);
  255. }
  256. if(tmp & HP_SDC_ACT_CALLBACK) {
  257. /* Note this means that irqhooks may be called
  258.  * in tasklet/bh context.
  259.  */
  260. if (curr->act.irqhook) 
  261. curr->act.irqhook(0, 0, 0, 0);
  262. }
  263. curr->actidx = curr->idx;
  264. curr->idx++;
  265. hp_sdc.rcurr = -1; 
  266. }
  267. }
  268. write_unlock_irq(&hp_sdc.rtq_lock);
  269. hp_sdc_put();
  270. }
  271. unsigned long hp_sdc_put(void) {
  272. hp_sdc_transaction *curr;
  273. uint8_t act;
  274. int idx, curridx;
  275. int limit = 0;
  276. write_lock(&hp_sdc.lock);
  277. /* If i8042 buffers are full, we cannot do anything that
  278.    requires output, so we skip to the administrativa. */
  279. if (hp_sdc.ibf) {
  280. hp_sdc_status_in8();
  281. if (hp_sdc.ibf) goto finish;
  282. }
  283.  anew:
  284. /* See if we are in the middle of a sequence. */
  285. if (hp_sdc.wcurr < 0) hp_sdc.wcurr = 0;
  286. read_lock_irq(&hp_sdc.rtq_lock);
  287. if (hp_sdc.rcurr == hp_sdc.wcurr) hp_sdc.wcurr++;
  288. read_unlock_irq(&hp_sdc.rtq_lock);
  289. if (hp_sdc.wcurr >= HP_SDC_QUEUE_LEN) hp_sdc.wcurr = 0;
  290. curridx = hp_sdc.wcurr;
  291. if (hp_sdc.tq[curridx] != NULL) goto start;
  292. while (++curridx != hp_sdc.wcurr) {
  293. if (curridx >= HP_SDC_QUEUE_LEN) {
  294. curridx = -1; /* Wrap to top */
  295. continue;
  296. }
  297. read_lock_irq(&hp_sdc.rtq_lock);
  298. if (hp_sdc.rcurr == curridx) {
  299. read_unlock_irq(&hp_sdc.rtq_lock);
  300. continue;
  301. }
  302. read_unlock_irq(&hp_sdc.rtq_lock);
  303. if (hp_sdc.tq[curridx] != NULL) break; /* Found one. */
  304. }
  305. if (curridx == hp_sdc.wcurr) { /* There's nothing queued to do. */
  306. curridx = -1;
  307. }
  308. hp_sdc.wcurr = curridx;
  309.  start:
  310. /* Check to see if the interrupt mask needs to be set. */
  311. if (hp_sdc.set_im) {
  312. hp_sdc_status_out8(hp_sdc.im | HP_SDC_CMD_SET_IM);
  313. hp_sdc.set_im = 0;
  314. goto finish;
  315. }
  316. if (hp_sdc.wcurr == -1) goto done;
  317. curr = hp_sdc.tq[curridx];
  318. idx = curr->actidx;
  319. if (curr->actidx >= curr->endidx) {
  320. hp_sdc.tq[curridx] = NULL;
  321. /* Interleave outbound data between the transactions. */
  322. hp_sdc.wcurr++;
  323. if (hp_sdc.wcurr >= HP_SDC_QUEUE_LEN) hp_sdc.wcurr = 0;
  324. goto finish;
  325. }
  326. act = curr->seq[idx];
  327. idx++;
  328. if (curr->idx >= curr->endidx) {
  329. if (act & HP_SDC_ACT_DEALLOC) kfree(curr);
  330. hp_sdc.tq[curridx] = NULL;
  331. /* Interleave outbound data between the transactions. */
  332. hp_sdc.wcurr++;
  333. if (hp_sdc.wcurr >= HP_SDC_QUEUE_LEN) hp_sdc.wcurr = 0;
  334. goto finish;
  335. }
  336. while (act & HP_SDC_ACT_PRECMD) {
  337. if (curr->idx != idx) {
  338. idx++;
  339. act &= ~HP_SDC_ACT_PRECMD;
  340. break;
  341. }
  342. hp_sdc_status_out8(curr->seq[idx]);
  343. curr->idx++;
  344. /* act finished? */
  345. if ((act & HP_SDC_ACT_DURING) == HP_SDC_ACT_PRECMD)
  346.   goto actdone;
  347. /* skip quantity field if data-out sequence follows. */
  348. if (act & HP_SDC_ACT_DATAOUT) curr->idx++;
  349. goto finish;
  350. }
  351. if (act & HP_SDC_ACT_DATAOUT) {
  352. int qty;
  353. qty = curr->seq[idx];
  354. idx++;
  355. if (curr->idx - idx < qty) {
  356. hp_sdc_data_out8(curr->seq[curr->idx]);
  357. curr->idx++;
  358. /* act finished? */
  359. if ((curr->idx - idx >= qty) && 
  360.     ((act & HP_SDC_ACT_DURING) == HP_SDC_ACT_DATAOUT))
  361. goto actdone;
  362. goto finish;
  363. }
  364. idx += qty;
  365. act &= ~HP_SDC_ACT_DATAOUT;
  366. }
  367. else while (act & HP_SDC_ACT_DATAREG) {
  368. int mask;
  369. uint8_t w7[4];
  370. mask = curr->seq[idx];
  371. if (idx != curr->idx) {
  372. idx++;
  373. idx += !!(mask & 1);
  374. idx += !!(mask & 2);
  375. idx += !!(mask & 4);
  376. idx += !!(mask & 8);
  377. act &= ~HP_SDC_ACT_DATAREG;
  378. break;
  379. }
  380. w7[0] = (mask & 1) ? curr->seq[++idx] : hp_sdc.r7[0];
  381. w7[1] = (mask & 2) ? curr->seq[++idx] : hp_sdc.r7[1];
  382. w7[2] = (mask & 4) ? curr->seq[++idx] : hp_sdc.r7[2];
  383. w7[3] = (mask & 8) ? curr->seq[++idx] : hp_sdc.r7[3];
  384. if (hp_sdc.wi > 0x73 || hp_sdc.wi < 0x70 ||
  385.         w7[hp_sdc.wi-0x70] == hp_sdc.r7[hp_sdc.wi-0x70]) {
  386. int i = 0;
  387. /* Need to point the write index register */
  388. while ((i < 4) && w7[i] == hp_sdc.r7[i]) i++;
  389. if (i < 4) {
  390. hp_sdc_status_out8(HP_SDC_CMD_SET_D0 + i);
  391. hp_sdc.wi = 0x70 + i;
  392. goto finish;
  393. }
  394. idx++;
  395. if ((act & HP_SDC_ACT_DURING) == HP_SDC_ACT_DATAREG)
  396. goto actdone;
  397. curr->idx = idx;
  398. act &= ~HP_SDC_ACT_DATAREG;
  399. break;
  400. }
  401. hp_sdc_data_out8(w7[hp_sdc.wi - 0x70]);
  402. hp_sdc.r7[hp_sdc.wi - 0x70] = w7[hp_sdc.wi - 0x70];
  403. hp_sdc.wi++; /* write index register autoincrements */
  404. {
  405. int i = 0;
  406. while ((i < 4) && w7[i] == hp_sdc.r7[i]) i++;
  407. if (i >= 4) {
  408. curr->idx = idx + 1;
  409. if ((act & HP_SDC_ACT_DURING) == 
  410.     HP_SDC_ACT_DATAREG)
  411.         goto actdone;
  412. }
  413. }
  414. goto finish;
  415. }
  416. /* We don't go any further in the command if there is a pending read,
  417.    because we don't want interleaved results. */
  418. read_lock_irq(&hp_sdc.rtq_lock);
  419. if (hp_sdc.rcurr >= 0) {
  420. read_unlock_irq(&hp_sdc.rtq_lock);
  421. goto finish;
  422. }
  423. read_unlock_irq(&hp_sdc.rtq_lock);
  424. if (act & HP_SDC_ACT_POSTCMD) {
  425.    uint8_t postcmd;
  426. /* curr->idx should == idx at this point. */
  427. postcmd = curr->seq[idx];
  428. curr->idx++;
  429. if (act & HP_SDC_ACT_DATAIN) {
  430. /* Start a new read */
  431.    hp_sdc.rqty = curr->seq[curr->idx];
  432. do_gettimeofday(&hp_sdc.rtv);
  433. curr->idx++;
  434. /* Still need to lock here in case of spurious irq. */
  435. write_lock_irq(&hp_sdc.rtq_lock);
  436. hp_sdc.rcurr = curridx; 
  437. write_unlock_irq(&hp_sdc.rtq_lock);
  438. hp_sdc_status_out8(postcmd);
  439. goto finish;
  440. }
  441. hp_sdc_status_out8(postcmd);
  442. goto actdone;
  443. }
  444. actdone:
  445. if (act & HP_SDC_ACT_SEMAPHORE) {
  446. up(curr->act.semaphore);
  447. }
  448. else if (act & HP_SDC_ACT_CALLBACK) {
  449. curr->act.irqhook(0,0,0,0);
  450. }
  451. if (curr->idx >= curr->endidx) { /* This transaction is over. */
  452. if (act & HP_SDC_ACT_DEALLOC) kfree(curr);
  453. hp_sdc.tq[curridx] = NULL;
  454. }
  455. else {
  456. curr->actidx = idx + 1;
  457. curr->idx = idx + 2;
  458. }
  459. /* Interleave outbound data between the transactions. */
  460. hp_sdc.wcurr++;
  461. if (hp_sdc.wcurr >= HP_SDC_QUEUE_LEN) hp_sdc.wcurr = 0;
  462.  finish:
  463. /* If by some quirk IBF has cleared and our ISR has run to 
  464.    see that that has happened, do it all again. */
  465. if (!hp_sdc.ibf && limit++ < 20) goto anew;
  466.  done:
  467. if (hp_sdc.wcurr >= 0) tasklet_schedule(&hp_sdc.task);
  468. write_unlock(&hp_sdc.lock);
  469. return 0;
  470. }
  471. /******* Functions called in either user or kernel context ****/
  472. int hp_sdc_enqueue_transaction(hp_sdc_transaction *this) {
  473. int i;
  474. if (this == NULL) {
  475. tasklet_schedule(&hp_sdc.task);
  476. return -EINVAL;
  477. };
  478. write_lock_bh(&hp_sdc.lock);
  479. /* Can't have same transaction on queue twice */
  480. for (i=0; i < HP_SDC_QUEUE_LEN; i++)
  481. if (hp_sdc.tq[i] == this) goto fail;
  482. this->actidx = 0;
  483. this->idx = 1;
  484. /* Search for empty slot */
  485. for (i=0; i < HP_SDC_QUEUE_LEN; i++) {
  486. if (hp_sdc.tq[i] == NULL) {
  487. hp_sdc.tq[i] = this;
  488. tasklet_schedule(&hp_sdc.task);
  489. write_unlock_bh(&hp_sdc.lock);
  490. return 0;
  491. }
  492. }
  493. write_unlock_bh(&hp_sdc.lock);
  494. printk(KERN_WARNING PREFIX "No free slot to add transaction.n");
  495. return -EBUSY;
  496.  fail:
  497. write_unlock_bh(&hp_sdc.lock);
  498. printk(KERN_WARNING PREFIX "Transaction add failed: transaction already queued?n");
  499. return -EINVAL;
  500. }
  501. int hp_sdc_dequeue_transaction(hp_sdc_transaction *this) {
  502. int i;
  503. write_lock_bh(&hp_sdc.lock);
  504. /* TODO: don't remove it if it's not done. */
  505. for (i=0; i < HP_SDC_QUEUE_LEN; i++)
  506. if (hp_sdc.tq[i] == this) hp_sdc.tq[i] = NULL;
  507. write_unlock_bh(&hp_sdc.lock);
  508. return 0;
  509. }
  510. /********************** User context functions **************************/
  511. int hp_sdc_request_timer_irq(hp_sdc_irqhook *callback) {
  512. MOD_INC_USE_COUNT;
  513. if (callback == NULL || hp_sdc.dev == NULL) {
  514. MOD_DEC_USE_COUNT;
  515. return -EINVAL;
  516. }
  517. write_lock_irq(&hp_sdc.hook_lock);
  518. if (hp_sdc.timer != NULL) {
  519. write_unlock_irq(&hp_sdc.hook_lock);
  520. MOD_DEC_USE_COUNT;
  521. return -EBUSY;
  522. }
  523. hp_sdc.timer = callback;
  524. /* Enable interrupts from the timers */
  525. hp_sdc.im &= ~HP_SDC_IM_FH;
  526.         hp_sdc.im &= ~HP_SDC_IM_PT;
  527. hp_sdc.im &= ~HP_SDC_IM_TIMERS;
  528. hp_sdc.set_im = 1;
  529. write_unlock_irq(&hp_sdc.hook_lock);
  530. tasklet_schedule(&hp_sdc.task);
  531. return 0;
  532. }
  533. int hp_sdc_request_hil_irq(hp_sdc_irqhook *callback) {
  534. MOD_INC_USE_COUNT;
  535. if (callback == NULL || hp_sdc.dev == NULL) {
  536. MOD_DEC_USE_COUNT;
  537. return -EINVAL;
  538. }
  539. write_lock_irq(&hp_sdc.hook_lock);
  540. if (hp_sdc.hil != NULL) {
  541. write_unlock_irq(&hp_sdc.hook_lock);
  542. MOD_DEC_USE_COUNT;
  543. return -EBUSY;
  544. }
  545. hp_sdc.hil = callback;
  546. hp_sdc.im &= ~HP_SDC_IM_HIL;
  547. hp_sdc.set_im = 1;
  548. write_unlock_irq(&hp_sdc.hook_lock);
  549. tasklet_schedule(&hp_sdc.task);
  550. return 0;
  551. }
  552. int hp_sdc_request_cooked_irq(hp_sdc_irqhook *callback) {
  553. MOD_INC_USE_COUNT;
  554. if (callback == NULL || hp_sdc.dev == NULL) {
  555. MOD_DEC_USE_COUNT;
  556. return -EINVAL;
  557. }
  558. write_lock_irq(&hp_sdc.hook_lock);
  559. if (hp_sdc.cooked != NULL) {
  560. write_unlock_irq(&hp_sdc.hook_lock);
  561. MOD_DEC_USE_COUNT;
  562. return -EBUSY;
  563. }
  564. /* Enable interrupts from the HIL MLC */
  565. hp_sdc.cooked = callback;
  566. hp_sdc.im &= ~HP_SDC_IM_HIL;
  567. hp_sdc.set_im = 1;
  568. write_unlock_irq(&hp_sdc.hook_lock);
  569. tasklet_schedule(&hp_sdc.task);
  570. return 0;
  571. }
  572. int hp_sdc_release_timer_irq(hp_sdc_irqhook *callback) {
  573. write_lock_irq(&hp_sdc.hook_lock);
  574. if ((callback != hp_sdc.timer) ||
  575.     (hp_sdc.timer == NULL)) {
  576. write_unlock_irq(&hp_sdc.hook_lock);
  577. return -EINVAL;
  578. }
  579. /* Disable interrupts from the timers */
  580. hp_sdc.timer = NULL;
  581. hp_sdc.im |= HP_SDC_IM_TIMERS;
  582. hp_sdc.im |= HP_SDC_IM_FH;
  583. hp_sdc.im |= HP_SDC_IM_PT;
  584. hp_sdc.set_im = 1;
  585. write_unlock_irq(&hp_sdc.hook_lock);
  586. tasklet_schedule(&hp_sdc.task);
  587. MOD_DEC_USE_COUNT;
  588. return 0;
  589. }
  590. int hp_sdc_release_hil_irq(hp_sdc_irqhook *callback) {
  591. write_lock_irq(&hp_sdc.hook_lock);
  592. if ((callback != hp_sdc.hil) ||
  593.     (hp_sdc.hil == NULL)) {
  594. write_unlock_irq(&hp_sdc.hook_lock);
  595. return -EINVAL;
  596. }
  597. hp_sdc.hil = NULL;
  598. /* Disable interrupts from HIL only if there is no cooked driver. */
  599. if(hp_sdc.cooked == NULL) {
  600. hp_sdc.im |= HP_SDC_IM_HIL;
  601. hp_sdc.set_im = 1;
  602. }
  603. write_unlock_irq(&hp_sdc.hook_lock);
  604. tasklet_schedule(&hp_sdc.task);
  605. MOD_DEC_USE_COUNT;
  606. return 0;
  607. }
  608. int hp_sdc_release_cooked_irq(hp_sdc_irqhook *callback) {
  609. write_lock_irq(&hp_sdc.hook_lock);
  610. if ((callback != hp_sdc.cooked) ||
  611.     (hp_sdc.cooked == NULL)) {
  612. write_unlock_irq(&hp_sdc.hook_lock);
  613. return -EINVAL;
  614. }
  615. hp_sdc.cooked = NULL;
  616. /* Disable interrupts from HIL only if there is no raw HIL driver. */
  617. if(hp_sdc.hil == NULL) {
  618. hp_sdc.im |= HP_SDC_IM_HIL;
  619. hp_sdc.set_im = 1;
  620. }
  621. write_unlock_irq(&hp_sdc.hook_lock);
  622. tasklet_schedule(&hp_sdc.task);
  623. MOD_DEC_USE_COUNT;
  624. return 0;
  625. }
  626. /************************* Keepalive timer task *********************/
  627. void hp_sdc_kicker (unsigned long data) {
  628. tasklet_schedule(&hp_sdc.task);
  629. /* Re-insert the periodic task. */
  630. mod_timer(&hp_sdc.kicker, jiffies + HZ);
  631. }
  632. /************************** Module Initialization ***************************/
  633. static struct parisc_device_id hp_sdc_tbl[] = {
  634. {
  635. hw_type: HPHW_FIO, 
  636. hversion_rev: HVERSION_REV_ANY_ID,
  637. hversion: HVERSION_ANY_ID,
  638. sversion: 0x73, 
  639.  },
  640. { 0, }
  641. };
  642. MODULE_DEVICE_TABLE(parisc, hp_sdc_tbl);
  643. static int __init hp_sdc_init(struct parisc_device *d);
  644. static struct parisc_driver hp_sdc_driver = {
  645. name: "HP SDC",
  646. id_table: hp_sdc_tbl,
  647. probe: hp_sdc_init,
  648. };
  649. static int __init hp_sdc_init(struct parisc_device *d)
  650. {
  651. int i;
  652. char *errstr = NULL;
  653. hp_sdc_transaction t_sync;
  654. uint8_t ts_sync[6];
  655. struct semaphore s_sync;
  656. errstr = "foon";
  657. if (!d) return 1;
  658. if (hp_sdc.dev != NULL) return 1; /* We only expect one SDC */
  659. hp_sdc.dev = d;
  660. hp_sdc.irq = d->irq;
  661. /* TODO: Is NMI == IRQ - 1 all cases, or is there a way to query? */
  662. hp_sdc.nmi = d->irq - 1;
  663. hp_sdc.base_io = (unsigned long) d->hpa;
  664. hp_sdc.data_io = (unsigned long) d->hpa + 0x800;
  665. hp_sdc.status_io = (unsigned long) d->hpa + 0x801;
  666.    hp_sdc.lock = RW_LOCK_UNLOCKED;
  667.    hp_sdc.ibf_lock = RW_LOCK_UNLOCKED;
  668.    hp_sdc.rtq_lock = RW_LOCK_UNLOCKED;
  669.    hp_sdc.hook_lock = RW_LOCK_UNLOCKED;
  670. hp_sdc.timer = NULL;
  671. hp_sdc.hil = NULL;
  672. hp_sdc.pup = NULL;
  673. hp_sdc.cooked = NULL;
  674. hp_sdc.im = HP_SDC_IM_MASK;  /* Mask maskable irqs */
  675. hp_sdc.set_im = 1;
  676. hp_sdc.wi = 0xff;
  677. hp_sdc.r7[0] = 0xff;
  678. hp_sdc.r7[1] = 0xff;
  679. hp_sdc.r7[2] = 0xff;
  680. hp_sdc.r7[3] = 0xff;
  681. hp_sdc.ibf = 1;
  682. for (i = 0; i < HP_SDC_QUEUE_LEN; i++) hp_sdc.tq[i] = NULL;
  683. hp_sdc.wcurr = -1;
  684.         hp_sdc.rcurr = -1;
  685. hp_sdc.rqty = 0;
  686. hp_sdc.dev_err = -ENODEV;
  687. errstr = "IO not found for";
  688. if (!hp_sdc.base_io) goto err0;
  689. errstr = "IRQ not found for";
  690. if (!hp_sdc.irq) goto err0;
  691. hp_sdc.dev_err = -EBUSY;
  692. errstr = "IO not available for";
  693.         if (request_region(hp_sdc.data_io, 2, hp_sdc_driver.name)) goto err0;
  694. errstr = "IRQ not available for";
  695.         if(request_irq(hp_sdc.irq, &hp_sdc_isr, 0, hp_sdc_driver.name, 
  696.        (void *) hp_sdc.base_io)) goto err1;
  697. errstr = "NMI not available for";
  698.         if (request_irq(hp_sdc.nmi, &hp_sdc_nmisr, 0, "HP SDC NMI", 
  699. (void*)d->hpa)) goto err2;
  700. printk(KERN_INFO PREFIX "HP SDC at 0x%p, IRQ %d (NMI IRQ %d)n", 
  701.        (void *)hp_sdc.base_io, hp_sdc.irq, hp_sdc.nmi);
  702. hp_sdc_status_in8();
  703. hp_sdc_data_in8();
  704. tasklet_init(&hp_sdc.task, hp_sdc_tasklet, 0);
  705. /* Sync the output buffer registers, thus scheduling hp_sdc_tasklet. */
  706. t_sync.actidx = 0;
  707. t_sync.idx = 1;
  708. t_sync.endidx = 6;
  709. t_sync.seq = ts_sync;
  710. ts_sync[0] = HP_SDC_ACT_DATAREG | HP_SDC_ACT_SEMAPHORE;
  711. ts_sync[1] = 0x0f;
  712. ts_sync[2] = ts_sync[3] = ts_sync[4] = ts_sync[5] = 0;
  713. t_sync.act.semaphore = &s_sync;
  714. init_MUTEX_LOCKED(&s_sync);
  715. hp_sdc_enqueue_transaction(&t_sync);
  716. down(&s_sync); /* Wait for t_sync to complete */
  717. /* Create the keepalive task */
  718. init_timer(&hp_sdc.kicker);
  719. hp_sdc.kicker.expires = jiffies + HZ;
  720. hp_sdc.kicker.function = &hp_sdc_kicker;
  721. add_timer(&hp_sdc.kicker);
  722. hp_sdc.dev_err = 0;
  723. return 0;
  724.  err2:
  725. free_irq(hp_sdc.irq, NULL);
  726.  err1:
  727. release_region(hp_sdc.data_io, 2);
  728.  err0:
  729. printk(KERN_WARNING PREFIX ": %s SDC IO=0x%p IRQ=0x%x NMI=0x%xn", 
  730. errstr, (void *)hp_sdc.base_io, hp_sdc.irq, hp_sdc.nmi);
  731. hp_sdc.dev = NULL;
  732. return hp_sdc.dev_err;
  733. }
  734. static void __exit hp_sdc_exit(void)
  735. {
  736. write_lock_irq(&hp_sdc.lock);
  737. /* Turn off all maskable "sub-function" irq's. */
  738. hp_sdc_spin_ibf();
  739. gsc_writeb(HP_SDC_CMD_SET_IM | HP_SDC_IM_MASK, hp_sdc.status_io);
  740. /* Wait until we know this has been processed by the i8042 */
  741. hp_sdc_spin_ibf();
  742. free_irq(hp_sdc.nmi, NULL);
  743. free_irq(hp_sdc.irq, NULL);
  744. write_unlock_irq(&hp_sdc.lock);
  745. del_timer(&hp_sdc.kicker);
  746. tasklet_kill(&hp_sdc.task);
  747. /*        release_region(hp_sdc.data_io, 2); */
  748. if (unregister_parisc_driver(&hp_sdc_driver)) 
  749. printk(KERN_WARNING PREFIX "Error unregistering HP SDC");
  750. }
  751. static int __init hp_sdc_register(void)
  752. {
  753. hp_sdc.dev = NULL;
  754. hp_sdc.dev_err = 0;
  755. hp_sdc_transaction tq_init;
  756. uint8_t tq_init_seq[5];
  757. struct semaphore tq_init_sem;
  758. if (register_parisc_driver(&hp_sdc_driver)) {
  759. printk(KERN_WARNING PREFIX "Error registering SDC with system bus tree.n");
  760. return -ENODEV;
  761. }
  762. if (hp_sdc.dev == NULL) {
  763. printk(KERN_WARNING PREFIX "No SDC found.n");
  764. return hp_sdc.dev_err;
  765. }
  766. init_MUTEX_LOCKED(&tq_init_sem);
  767. tq_init.actidx = 0;
  768. tq_init.idx = 1;
  769. tq_init.endidx = 5;
  770. tq_init.seq = tq_init_seq;
  771. tq_init.act.semaphore = &tq_init_sem;
  772. tq_init_seq[0] = 
  773.   HP_SDC_ACT_POSTCMD | HP_SDC_ACT_DATAIN | HP_SDC_ACT_SEMAPHORE;
  774. tq_init_seq[1] = HP_SDC_CMD_READ_KCC;
  775. tq_init_seq[2] = 1;
  776. tq_init_seq[3] = 0;
  777. tq_init_seq[4] = 0;
  778. hp_sdc_enqueue_transaction(&tq_init);
  779. down(&tq_init_sem);
  780. up(&tq_init_sem);
  781. if ((tq_init_seq[0] & HP_SDC_ACT_DEAD) == HP_SDC_ACT_DEAD) {
  782. printk(KERN_WARNING PREFIX "Error reading config byte.n");
  783. hp_sdc_exit();
  784. return -ENODEV;
  785. }
  786. hp_sdc.r11 = tq_init_seq[4];
  787. if (hp_sdc.r11 & HP_SDC_CFG_NEW) {
  788. char *str;
  789. printk(KERN_INFO PREFIX "New style SDCn");
  790. tq_init_seq[1] = HP_SDC_CMD_READ_XTD;
  791. tq_init.actidx = 0;
  792. tq_init.idx = 1;
  793. down(&tq_init_sem);
  794. hp_sdc_enqueue_transaction(&tq_init);
  795. down(&tq_init_sem);
  796. up(&tq_init_sem);
  797. if ((tq_init_seq[0] & HP_SDC_ACT_DEAD) == HP_SDC_ACT_DEAD) {
  798. printk(KERN_WARNING PREFIX "Error reading extended config byte.n");
  799. return -ENODEV;
  800. }
  801. hp_sdc.r7e = tq_init_seq[4];
  802. HP_SDC_XTD_REV_STRINGS(hp_sdc.r7e & HP_SDC_XTD_REV, str)
  803. printk(KERN_INFO PREFIX "Revision: %sn", str);
  804. if (hp_sdc.r7e & HP_SDC_XTD_BEEPER) {
  805. printk(KERN_INFO PREFIX "TI SN76494 beeper presentn");
  806. }
  807. if (hp_sdc.r7e & HP_SDC_XTD_BBRTC) {
  808. printk(KERN_INFO PREFIX "OKI MSM-58321 BBRTC presentn");
  809. }
  810. printk(KERN_INFO PREFIX "Spunking the self test register to force PUP "
  811.        "on next firmware reset.n");
  812. tq_init_seq[0] = HP_SDC_ACT_PRECMD | 
  813. HP_SDC_ACT_DATAOUT | HP_SDC_ACT_SEMAPHORE;
  814. tq_init_seq[1] = HP_SDC_CMD_SET_STR;
  815. tq_init_seq[2] = 1;
  816. tq_init_seq[3] = 0;
  817. tq_init.actidx = 0;
  818. tq_init.idx = 1;
  819. tq_init.endidx = 4;
  820. down(&tq_init_sem);
  821. hp_sdc_enqueue_transaction(&tq_init);
  822. down(&tq_init_sem);
  823. up(&tq_init_sem);
  824. }
  825. else {
  826. printk(KERN_INFO PREFIX "Old style SDC (1820-%s).n", 
  827.        (hp_sdc.r11 & HP_SDC_CFG_REV) ? "3300" : "2564/3087");
  828. }
  829.         return 0;
  830. }
  831. module_init(hp_sdc_register);
  832. module_exit(hp_sdc_exit);
  833. /* Timing notes:  These measurements taken on my 64MHz 7100-LC (715/64) 
  834.  *                                              cycles cycles-adj    time
  835.  * between two consecutive mfctl(16)'s:              4        n/a    63ns
  836.  * hp_sdc_spin_ibf when idle:                      119        115   1.7us
  837.  * gsc_writeb status register:                      83         79   1.2us
  838.  * IBF to clear after sending SET_IM:             6204       6006    93us
  839.  * IBF to clear after sending LOAD_RT:            4467       4352    68us  
  840.  * IBF to clear after sending two LOAD_RTs:      18974      18859   295us
  841.  * READ_T1, read status/data, IRQ, call handler: 35564        n/a   556us
  842.  * cmd to ~IBF READ_T1 2nd time right after:   5158403        n/a    81ms
  843.  * between IRQ received and ~IBF for above:    2578877        n/a    40ms
  844.  *
  845.  * Performance stats after a run of this module configuring HIL and
  846.  * receiving a few mouse events:
  847.  *
  848.  * status in8  282508 cycles 7128 calls
  849.  * status out8   8404 cycles  341 calls
  850.  * data out8     1734 cycles   78 calls
  851.  * isr         174324 cycles  617 calls (includes take)
  852.  * take          1241 cycles    2 calls
  853.  * put        1411504 cycles 6937 calls
  854.  * task       1655209 cycles 6937 calls (includes put)
  855.  *
  856.  */