mixart_core.c 18 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588
  1. /*
  2. * Driver for Digigram miXart soundcards
  3. *
  4. * low level interface with interrupt handling and mail box implementation
  5. *
  6. * Copyright (c) 2003 by Digigram <alsa@digigram.com>
  7. *
  8. * This program is free software; you can redistribute it and/or modify
  9. * it under the terms of the GNU General Public License as published by
  10. * the Free Software Foundation; either version 2 of the License, or
  11. * (at your option) any later version.
  12. *
  13. * This program is distributed in the hope that it will be useful,
  14. * but WITHOUT ANY WARRANTY; without even the implied warranty of
  15. * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  16. * GNU General Public License for more details.
  17. *
  18. * You should have received a copy of the GNU General Public License
  19. * along with this program; if not, write to the Free Software
  20. * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
  21. */
  22. #include <sound/driver.h>
  23. #include <linux/interrupt.h>
  24. #include <asm/io.h>
  25. #include <sound/core.h>
  26. #include "mixart.h"
  27. #include "mixart_hwdep.h"
  28. #include "mixart_core.h"
  29. #define MSG_TIMEOUT_JIFFIES (400 * HZ) / 1000 /* 400 ms */
  30. #define MSG_DESCRIPTOR_SIZE 0x24
  31. #define MSG_HEADER_SIZE (MSG_DESCRIPTOR_SIZE + 4)
  32. #define MSG_DEFAULT_SIZE 512
  33. #define MSG_TYPE_MASK 0x00000003 /* mask for following types */
  34. #define MSG_TYPE_NOTIFY 0 /* embedded -> driver (only notification, do not get_msg() !) */
  35. #define MSG_TYPE_COMMAND 1 /* driver <-> embedded (a command has no answer) */
  36. #define MSG_TYPE_REQUEST 2 /* driver -> embedded (request will get an answer back) */
  37. #define MSG_TYPE_ANSWER 3 /* embedded -> driver */
  38. #define MSG_CANCEL_NOTIFY_MASK 0x80000000 /* this bit is set for a notification that has been canceled */
  39. static int retrieve_msg_frame(mixart_mgr_t *mgr, u32 *msg_frame)
  40. {
  41. /* read the message frame fifo */
  42. u32 headptr, tailptr;
  43. tailptr = readl_be(MIXART_MEM(mgr, MSG_OUTBOUND_POST_TAIL));
  44. headptr = readl_be(MIXART_MEM(mgr, MSG_OUTBOUND_POST_HEAD));
  45. if (tailptr == headptr)
  46. return 0; /* no message posted */
  47. snd_assert( tailptr >= MSG_OUTBOUND_POST_STACK, return 0); /* error */
  48. snd_assert( tailptr < (MSG_OUTBOUND_POST_STACK+MSG_BOUND_STACK_SIZE), return 0); /* error */
  49. *msg_frame = readl_be(MIXART_MEM(mgr, tailptr));
  50. /* increment the tail index */
  51. tailptr += 4;
  52. if( tailptr >= (MSG_OUTBOUND_POST_STACK+MSG_BOUND_STACK_SIZE) )
  53. tailptr = MSG_OUTBOUND_POST_STACK;
  54. writel_be(tailptr, MIXART_MEM(mgr, MSG_OUTBOUND_POST_TAIL));
  55. return 1;
  56. }
  57. static int get_msg(mixart_mgr_t *mgr, mixart_msg_t *resp, u32 msg_frame_address )
  58. {
  59. unsigned long flags;
  60. u32 headptr;
  61. u32 size;
  62. int err;
  63. #ifndef __BIG_ENDIAN
  64. unsigned int i;
  65. #endif
  66. spin_lock_irqsave(&mgr->msg_lock, flags);
  67. err = 0;
  68. /* copy message descriptor from miXart to driver */
  69. size = readl_be(MIXART_MEM(mgr, msg_frame_address)); /* size of descriptor + response */
  70. resp->message_id = readl_be(MIXART_MEM(mgr, msg_frame_address + 4)); /* dwMessageID */
  71. resp->uid.object_id = readl_be(MIXART_MEM(mgr, msg_frame_address + 8)); /* uidDest */
  72. resp->uid.desc = readl_be(MIXART_MEM(mgr, msg_frame_address + 12)); /* */
  73. if( (size < MSG_DESCRIPTOR_SIZE) || (resp->size < (size - MSG_DESCRIPTOR_SIZE))) {
  74. err = -EINVAL;
  75. snd_printk(KERN_ERR "problem with response size = %d\n", size);
  76. goto _clean_exit;
  77. }
  78. size -= MSG_DESCRIPTOR_SIZE;
  79. memcpy_fromio(resp->data, MIXART_MEM(mgr, msg_frame_address + MSG_HEADER_SIZE ), size);
  80. resp->size = size;
  81. /* swap if necessary */
  82. #ifndef __BIG_ENDIAN
  83. size /= 4; /* u32 size */
  84. for(i=0; i < size; i++) {
  85. ((u32*)resp->data)[i] = be32_to_cpu(((u32*)resp->data)[i]);
  86. }
  87. #endif
  88. /*
  89. * free message frame address
  90. */
  91. headptr = readl_be(MIXART_MEM(mgr, MSG_OUTBOUND_FREE_HEAD));
  92. if( (headptr < MSG_OUTBOUND_FREE_STACK) || ( headptr >= (MSG_OUTBOUND_FREE_STACK+MSG_BOUND_STACK_SIZE))) {
  93. err = -EINVAL;
  94. goto _clean_exit;
  95. }
  96. /* give address back to outbound fifo */
  97. writel_be(msg_frame_address, MIXART_MEM(mgr, headptr));
  98. /* increment the outbound free head */
  99. headptr += 4;
  100. if( headptr >= (MSG_OUTBOUND_FREE_STACK+MSG_BOUND_STACK_SIZE) )
  101. headptr = MSG_OUTBOUND_FREE_STACK;
  102. writel_be(headptr, MIXART_MEM(mgr, MSG_OUTBOUND_FREE_HEAD));
  103. _clean_exit:
  104. spin_unlock_irqrestore(&mgr->msg_lock, flags);
  105. return err;
  106. }
  107. /*
  108. * send a message to miXart. return: the msg_frame used for this message
  109. */
  110. /* call with mgr->msg_lock held! */
  111. static int send_msg( mixart_mgr_t *mgr,
  112. mixart_msg_t *msg,
  113. int max_answersize,
  114. int mark_pending,
  115. u32 *msg_event)
  116. {
  117. u32 headptr, tailptr;
  118. u32 msg_frame_address;
  119. int err, i;
  120. snd_assert(msg->size % 4 == 0, return -EINVAL);
  121. err = 0;
  122. /* get message frame address */
  123. tailptr = readl_be(MIXART_MEM(mgr, MSG_INBOUND_FREE_TAIL));
  124. headptr = readl_be(MIXART_MEM(mgr, MSG_INBOUND_FREE_HEAD));
  125. if (tailptr == headptr) {
  126. snd_printk(KERN_ERR "error: no message frame available\n");
  127. return -EBUSY;
  128. }
  129. if( (tailptr < MSG_INBOUND_FREE_STACK) || (tailptr >= (MSG_INBOUND_FREE_STACK+MSG_BOUND_STACK_SIZE))) {
  130. return -EINVAL;
  131. }
  132. msg_frame_address = readl_be(MIXART_MEM(mgr, tailptr));
  133. writel(0, MIXART_MEM(mgr, tailptr)); /* set address to zero on this fifo position */
  134. /* increment the inbound free tail */
  135. tailptr += 4;
  136. if( tailptr >= (MSG_INBOUND_FREE_STACK+MSG_BOUND_STACK_SIZE) )
  137. tailptr = MSG_INBOUND_FREE_STACK;
  138. writel_be(tailptr, MIXART_MEM(mgr, MSG_INBOUND_FREE_TAIL));
  139. /* TODO : use memcpy_toio() with intermediate buffer to copy the message */
  140. /* copy message descriptor to card memory */
  141. writel_be( msg->size + MSG_DESCRIPTOR_SIZE, MIXART_MEM(mgr, msg_frame_address) ); /* size of descriptor + request */
  142. writel_be( msg->message_id , MIXART_MEM(mgr, msg_frame_address + 4) ); /* dwMessageID */
  143. writel_be( msg->uid.object_id, MIXART_MEM(mgr, msg_frame_address + 8) ); /* uidDest */
  144. writel_be( msg->uid.desc, MIXART_MEM(mgr, msg_frame_address + 12) ); /* */
  145. writel_be( MSG_DESCRIPTOR_SIZE, MIXART_MEM(mgr, msg_frame_address + 16) ); /* SizeHeader */
  146. writel_be( MSG_DESCRIPTOR_SIZE, MIXART_MEM(mgr, msg_frame_address + 20) ); /* OffsetDLL_T16 */
  147. writel_be( msg->size, MIXART_MEM(mgr, msg_frame_address + 24) ); /* SizeDLL_T16 */
  148. writel_be( MSG_DESCRIPTOR_SIZE, MIXART_MEM(mgr, msg_frame_address + 28) ); /* OffsetDLL_DRV */
  149. writel_be( 0, MIXART_MEM(mgr, msg_frame_address + 32) ); /* SizeDLL_DRV */
  150. writel_be( MSG_DESCRIPTOR_SIZE + max_answersize, MIXART_MEM(mgr, msg_frame_address + 36) ); /* dwExpectedAnswerSize */
  151. /* copy message data to card memory */
  152. for( i=0; i < msg->size; i+=4 ) {
  153. writel_be( *(u32*)(msg->data + i), MIXART_MEM(mgr, MSG_HEADER_SIZE + msg_frame_address + i) );
  154. }
  155. if( mark_pending ) {
  156. if( *msg_event ) {
  157. /* the pending event is the notification we wait for ! */
  158. mgr->pending_event = *msg_event;
  159. }
  160. else {
  161. /* the pending event is the answer we wait for (same address than the request)! */
  162. mgr->pending_event = msg_frame_address;
  163. /* copy address back to caller */
  164. *msg_event = msg_frame_address;
  165. }
  166. }
  167. /* mark the frame as a request (will have an answer) */
  168. msg_frame_address |= MSG_TYPE_REQUEST;
  169. /* post the frame */
  170. headptr = readl_be(MIXART_MEM(mgr, MSG_INBOUND_POST_HEAD));
  171. if( (headptr < MSG_INBOUND_POST_STACK) || (headptr >= (MSG_INBOUND_POST_STACK+MSG_BOUND_STACK_SIZE))) {
  172. return -EINVAL;
  173. }
  174. writel_be(msg_frame_address, MIXART_MEM(mgr, headptr));
  175. /* increment the inbound post head */
  176. headptr += 4;
  177. if( headptr >= (MSG_INBOUND_POST_STACK+MSG_BOUND_STACK_SIZE) )
  178. headptr = MSG_INBOUND_POST_STACK;
  179. writel_be(headptr, MIXART_MEM(mgr, MSG_INBOUND_POST_HEAD));
  180. return 0;
  181. }
  182. int snd_mixart_send_msg(mixart_mgr_t *mgr, mixart_msg_t *request, int max_resp_size, void *resp_data)
  183. {
  184. mixart_msg_t resp;
  185. u32 msg_frame = 0; /* set to 0, so it's no notification to wait for, but the answer */
  186. int err;
  187. wait_queue_t wait;
  188. long timeout;
  189. down(&mgr->msg_mutex);
  190. init_waitqueue_entry(&wait, current);
  191. spin_lock_irq(&mgr->msg_lock);
  192. /* send the message */
  193. err = send_msg(mgr, request, max_resp_size, 1, &msg_frame); /* send and mark the answer pending */
  194. if (err) {
  195. spin_unlock_irq(&mgr->msg_lock);
  196. up(&mgr->msg_mutex);
  197. return err;
  198. }
  199. set_current_state(TASK_UNINTERRUPTIBLE);
  200. add_wait_queue(&mgr->msg_sleep, &wait);
  201. spin_unlock_irq(&mgr->msg_lock);
  202. timeout = schedule_timeout(MSG_TIMEOUT_JIFFIES);
  203. remove_wait_queue(&mgr->msg_sleep, &wait);
  204. if (! timeout) {
  205. /* error - no ack */
  206. up(&mgr->msg_mutex);
  207. snd_printk(KERN_ERR "error: no reponse on msg %x\n", msg_frame);
  208. return -EIO;
  209. }
  210. /* retrieve the answer into the same mixart_msg_t */
  211. resp.message_id = 0;
  212. resp.uid = (mixart_uid_t){0,0};
  213. resp.data = resp_data;
  214. resp.size = max_resp_size;
  215. err = get_msg(mgr, &resp, msg_frame);
  216. if( request->message_id != resp.message_id )
  217. snd_printk(KERN_ERR "REPONSE ERROR!\n");
  218. up(&mgr->msg_mutex);
  219. return err;
  220. }
  221. int snd_mixart_send_msg_wait_notif(mixart_mgr_t *mgr, mixart_msg_t *request, u32 notif_event)
  222. {
  223. int err;
  224. wait_queue_t wait;
  225. long timeout;
  226. snd_assert(notif_event != 0, return -EINVAL);
  227. snd_assert((notif_event & MSG_TYPE_MASK) == MSG_TYPE_NOTIFY, return -EINVAL);
  228. snd_assert((notif_event & MSG_CANCEL_NOTIFY_MASK) == 0, return -EINVAL);
  229. down(&mgr->msg_mutex);
  230. init_waitqueue_entry(&wait, current);
  231. spin_lock_irq(&mgr->msg_lock);
  232. /* send the message */
  233. err = send_msg(mgr, request, MSG_DEFAULT_SIZE, 1, &notif_event); /* send and mark the notification event pending */
  234. if(err) {
  235. spin_unlock_irq(&mgr->msg_lock);
  236. up(&mgr->msg_mutex);
  237. return err;
  238. }
  239. set_current_state(TASK_UNINTERRUPTIBLE);
  240. add_wait_queue(&mgr->msg_sleep, &wait);
  241. spin_unlock_irq(&mgr->msg_lock);
  242. timeout = schedule_timeout(MSG_TIMEOUT_JIFFIES);
  243. remove_wait_queue(&mgr->msg_sleep, &wait);
  244. if (! timeout) {
  245. /* error - no ack */
  246. up(&mgr->msg_mutex);
  247. snd_printk(KERN_ERR "error: notification %x not received\n", notif_event);
  248. return -EIO;
  249. }
  250. up(&mgr->msg_mutex);
  251. return 0;
  252. }
  253. int snd_mixart_send_msg_nonblock(mixart_mgr_t *mgr, mixart_msg_t *request)
  254. {
  255. u32 message_frame;
  256. unsigned long flags;
  257. int err;
  258. /* just send the message (do not mark it as a pending one) */
  259. spin_lock_irqsave(&mgr->msg_lock, flags);
  260. err = send_msg(mgr, request, MSG_DEFAULT_SIZE, 0, &message_frame);
  261. spin_unlock_irqrestore(&mgr->msg_lock, flags);
  262. /* the answer will be handled by snd_mixart_msg_tasklet() */
  263. atomic_inc(&mgr->msg_processed);
  264. return err;
  265. }
  266. /* common buffer of tasklet and interrupt to send/receive messages */
  267. static u32 mixart_msg_data[MSG_DEFAULT_SIZE / 4];
  268. void snd_mixart_msg_tasklet( unsigned long arg)
  269. {
  270. mixart_mgr_t *mgr = ( mixart_mgr_t*)(arg);
  271. mixart_msg_t resp;
  272. u32 msg, addr, type;
  273. int err;
  274. spin_lock(&mgr->lock);
  275. while (mgr->msg_fifo_readptr != mgr->msg_fifo_writeptr) {
  276. msg = mgr->msg_fifo[mgr->msg_fifo_readptr];
  277. mgr->msg_fifo_readptr++;
  278. mgr->msg_fifo_readptr %= MSG_FIFO_SIZE;
  279. /* process the message ... */
  280. addr = msg & ~MSG_TYPE_MASK;
  281. type = msg & MSG_TYPE_MASK;
  282. switch (type) {
  283. case MSG_TYPE_ANSWER:
  284. /* answer to a message on that we did not wait for (send_msg_nonblock) */
  285. resp.message_id = 0;
  286. resp.data = mixart_msg_data;
  287. resp.size = sizeof(mixart_msg_data);
  288. err = get_msg(mgr, &resp, addr);
  289. if( err < 0 ) {
  290. snd_printk(KERN_ERR "tasklet: error(%d) reading mf %x\n", err, msg);
  291. break;
  292. }
  293. switch(resp.message_id) {
  294. case MSG_STREAM_START_INPUT_STAGE_PACKET:
  295. case MSG_STREAM_START_OUTPUT_STAGE_PACKET:
  296. case MSG_STREAM_STOP_INPUT_STAGE_PACKET:
  297. case MSG_STREAM_STOP_OUTPUT_STAGE_PACKET:
  298. if(mixart_msg_data[0])
  299. snd_printk(KERN_ERR "tasklet : error MSG_STREAM_ST***_***PUT_STAGE_PACKET status=%x\n", mixart_msg_data[0]);
  300. break;
  301. default:
  302. snd_printdd("tasklet received mf(%x) : msg_id(%x) uid(%x, %x) size(%zd)\n",
  303. msg, resp.message_id, resp.uid.object_id, resp.uid.desc, resp.size);
  304. break;
  305. }
  306. break;
  307. case MSG_TYPE_NOTIFY:
  308. /* msg contains no address ! do not get_msg() ! */
  309. case MSG_TYPE_COMMAND:
  310. /* get_msg() necessary */
  311. default:
  312. snd_printk(KERN_ERR "tasklet doesn't know what to do with message %x\n", msg);
  313. } /* switch type */
  314. /* decrement counter */
  315. atomic_dec(&mgr->msg_processed);
  316. } /* while there is a msg in fifo */
  317. spin_unlock(&mgr->lock);
  318. }
  319. irqreturn_t snd_mixart_interrupt(int irq, void *dev_id, struct pt_regs *regs)
  320. {
  321. mixart_mgr_t *mgr = dev_id;
  322. int err;
  323. mixart_msg_t resp;
  324. u32 msg;
  325. u32 it_reg;
  326. spin_lock(&mgr->lock);
  327. it_reg = readl_le(MIXART_REG(mgr, MIXART_PCI_OMISR_OFFSET));
  328. if( !(it_reg & MIXART_OIDI) ) {
  329. /* this device did not cause the interrupt */
  330. spin_unlock(&mgr->lock);
  331. return IRQ_NONE;
  332. }
  333. /* mask all interrupts */
  334. writel_le(MIXART_HOST_ALL_INTERRUPT_MASKED, MIXART_REG(mgr, MIXART_PCI_OMIMR_OFFSET));
  335. /* outdoorbell register clear */
  336. it_reg = readl(MIXART_REG(mgr, MIXART_PCI_ODBR_OFFSET));
  337. writel(it_reg, MIXART_REG(mgr, MIXART_PCI_ODBR_OFFSET));
  338. /* clear interrupt */
  339. writel_le( MIXART_OIDI, MIXART_REG(mgr, MIXART_PCI_OMISR_OFFSET) );
  340. /* process interrupt */
  341. while (retrieve_msg_frame(mgr, &msg)) {
  342. switch (msg & MSG_TYPE_MASK) {
  343. case MSG_TYPE_COMMAND:
  344. resp.message_id = 0;
  345. resp.data = mixart_msg_data;
  346. resp.size = sizeof(mixart_msg_data);
  347. err = get_msg(mgr, &resp, msg & ~MSG_TYPE_MASK);
  348. if( err < 0 ) {
  349. snd_printk(KERN_ERR "interrupt: error(%d) reading mf %x\n", err, msg);
  350. break;
  351. }
  352. if(resp.message_id == MSG_SERVICES_TIMER_NOTIFY) {
  353. int i;
  354. mixart_timer_notify_t *notify = (mixart_timer_notify_t*)mixart_msg_data;
  355. for(i=0; i<notify->stream_count; i++) {
  356. u32 buffer_id = notify->streams[i].buffer_id;
  357. unsigned int chip_number = (buffer_id & MIXART_NOTIFY_CARD_MASK) >> MIXART_NOTIFY_CARD_OFFSET; /* card0 to 3 */
  358. unsigned int pcm_number = (buffer_id & MIXART_NOTIFY_PCM_MASK ) >> MIXART_NOTIFY_PCM_OFFSET; /* pcm0 to 3 */
  359. unsigned int sub_number = buffer_id & MIXART_NOTIFY_SUBS_MASK; /* 0 to MIXART_PLAYBACK_STREAMS */
  360. unsigned int is_capture = ((buffer_id & MIXART_NOTIFY_CAPT_MASK) != 0); /* playback == 0 / capture == 1 */
  361. mixart_t *chip = mgr->chip[chip_number];
  362. mixart_stream_t *stream;
  363. if ((chip_number >= mgr->num_cards) || (pcm_number >= MIXART_PCM_TOTAL) || (sub_number >= MIXART_PLAYBACK_STREAMS)) {
  364. snd_printk(KERN_ERR "error MSG_SERVICES_TIMER_NOTIFY buffer_id (%x) pos(%d)\n",
  365. buffer_id, notify->streams[i].sample_pos_low_part);
  366. break;
  367. }
  368. if (is_capture)
  369. stream = &chip->capture_stream[pcm_number];
  370. else
  371. stream = &chip->playback_stream[pcm_number][sub_number];
  372. if (stream->substream && (stream->status == MIXART_STREAM_STATUS_RUNNING)) {
  373. snd_pcm_runtime_t *runtime = stream->substream->runtime;
  374. int elapsed = 0;
  375. u64 sample_count = ((u64)notify->streams[i].sample_pos_high_part) << 32;
  376. sample_count |= notify->streams[i].sample_pos_low_part;
  377. while (1) {
  378. u64 new_elapse_pos = stream->abs_period_elapsed + runtime->period_size;
  379. if (new_elapse_pos > sample_count) {
  380. break; /* while */
  381. }
  382. else {
  383. elapsed = 1;
  384. stream->buf_periods++;
  385. if (stream->buf_periods >= runtime->periods)
  386. stream->buf_periods = 0;
  387. stream->abs_period_elapsed = new_elapse_pos;
  388. }
  389. }
  390. stream->buf_period_frag = (u32)( sample_count - stream->abs_period_elapsed );
  391. if(elapsed) {
  392. spin_unlock(&mgr->lock);
  393. snd_pcm_period_elapsed(stream->substream);
  394. spin_lock(&mgr->lock);
  395. }
  396. }
  397. }
  398. break;
  399. }
  400. if(resp.message_id == MSG_SERVICES_REPORT_TRACES) {
  401. if(resp.size > 1) {
  402. #ifndef __BIG_ENDIAN
  403. /* Traces are text: the swapped msg_data has to be swapped back ! */
  404. int i;
  405. for(i=0; i<(resp.size/4); i++) {
  406. (mixart_msg_data)[i] = cpu_to_be32((mixart_msg_data)[i]);
  407. }
  408. #endif
  409. ((char*)mixart_msg_data)[resp.size - 1] = 0;
  410. snd_printdd("MIXART TRACE : %s\n", (char*)mixart_msg_data);
  411. }
  412. break;
  413. }
  414. snd_printdd("command %x not handled\n", resp.message_id);
  415. break;
  416. case MSG_TYPE_NOTIFY:
  417. if(msg & MSG_CANCEL_NOTIFY_MASK) {
  418. msg &= ~MSG_CANCEL_NOTIFY_MASK;
  419. snd_printk(KERN_ERR "canceled notification %x !\n", msg);
  420. }
  421. /* no break, continue ! */
  422. case MSG_TYPE_ANSWER:
  423. /* answer or notification to a message we are waiting for*/
  424. spin_lock(&mgr->msg_lock);
  425. if( (msg & ~MSG_TYPE_MASK) == mgr->pending_event ) {
  426. wake_up(&mgr->msg_sleep);
  427. mgr->pending_event = 0;
  428. }
  429. /* answer to a message we did't want to wait for */
  430. else {
  431. mgr->msg_fifo[mgr->msg_fifo_writeptr] = msg;
  432. mgr->msg_fifo_writeptr++;
  433. mgr->msg_fifo_writeptr %= MSG_FIFO_SIZE;
  434. tasklet_hi_schedule(&mgr->msg_taskq);
  435. }
  436. spin_unlock(&mgr->msg_lock);
  437. break;
  438. case MSG_TYPE_REQUEST:
  439. default:
  440. snd_printdd("interrupt received request %x\n", msg);
  441. /* TODO : are there things to do here ? */
  442. break;
  443. } /* switch on msg type */
  444. } /* while there are msgs */
  445. /* allow interrupt again */
  446. writel_le( MIXART_ALLOW_OUTBOUND_DOORBELL, MIXART_REG( mgr, MIXART_PCI_OMIMR_OFFSET));
  447. spin_unlock(&mgr->lock);
  448. return IRQ_HANDLED;
  449. }
  450. void snd_mixart_init_mailbox(mixart_mgr_t *mgr)
  451. {
  452. writel( 0, MIXART_MEM( mgr, MSG_HOST_RSC_PROTECTION ) );
  453. writel( 0, MIXART_MEM( mgr, MSG_AGENT_RSC_PROTECTION ) );
  454. /* allow outbound messagebox to generate interrupts */
  455. if(mgr->irq >= 0) {
  456. writel_le( MIXART_ALLOW_OUTBOUND_DOORBELL, MIXART_REG( mgr, MIXART_PCI_OMIMR_OFFSET));
  457. }
  458. return;
  459. }
  460. void snd_mixart_exit_mailbox(mixart_mgr_t *mgr)
  461. {
  462. /* no more interrupts on outbound messagebox */
  463. writel_le( MIXART_HOST_ALL_INTERRUPT_MASKED, MIXART_REG( mgr, MIXART_PCI_OMIMR_OFFSET));
  464. return;
  465. }
  466. void snd_mixart_reset_board(mixart_mgr_t *mgr)
  467. {
  468. /* reset miXart */
  469. writel_be( 1, MIXART_REG(mgr, MIXART_BA1_BRUTAL_RESET_OFFSET) );
  470. return;
  471. }