1 /* ////////////////////////////////////////////////////////////////////////// */
3 /* Copyright (c) Atmel Corporation. All rights reserved. */
5 /* Module Name: wilc_sdio.c */
8 /* //////////////////////////////////////////////////////////////////////////// */
10 #include "wilc_wlan_if.h"
11 #include "wilc_wlan.h"
13 #define WILC_SDIO_BLOCK_SIZE 512
18 int (*sdio_cmd52)(sdio_cmd52_t *);
19 int (*sdio_cmd53)(sdio_cmd53_t *);
20 int (*sdio_set_max_speed)(void);
21 int (*sdio_set_default_speed)(void);
22 wilc_debug_func dPrint;
24 #define MAX_NUN_INT_THRPT_ENH2 (5) /* Max num interrupts allowed in registers 0xf7, 0xf8 */
28 static wilc_sdio_t g_sdio;
30 #ifdef WILC_SDIO_IRQ_GPIO
31 static int sdio_write_reg(u32 addr, u32 data);
32 static int sdio_read_reg(u32 addr, u32 *data);
35 /********************************************
39 ********************************************/
41 static int sdio_set_func0_csa_address(u32 adr)
53 if (!g_sdio.sdio_cmd52(&cmd)) {
54 g_sdio.dPrint(N_ERR, "[wilc sdio]: Failed cmd52, set 0x10c data...\n");
59 cmd.data = (u8)(adr >> 8);
60 if (!g_sdio.sdio_cmd52(&cmd)) {
61 g_sdio.dPrint(N_ERR, "[wilc sdio]: Failed cmd52, set 0x10d data...\n");
66 cmd.data = (u8)(adr >> 16);
67 if (!g_sdio.sdio_cmd52(&cmd)) {
68 g_sdio.dPrint(N_ERR, "[wilc sdio]: Failed cmd52, set 0x10e data...\n");
77 static int sdio_set_func0_csa_address_byte0(u32 adr)
89 if (!g_sdio.sdio_cmd52(&cmd)) {
90 g_sdio.dPrint(N_ERR, "[wilc sdio]: Failed cmd52, set 0x10c data...\n");
99 static int sdio_set_func0_block_size(u32 block_size)
107 cmd.data = (u8)block_size;
108 if (!g_sdio.sdio_cmd52(&cmd)) {
109 g_sdio.dPrint(N_ERR, "[wilc sdio]: Failed cmd52, set 0x10 data...\n");
114 cmd.data = (u8)(block_size >> 8);
115 if (!g_sdio.sdio_cmd52(&cmd)) {
116 g_sdio.dPrint(N_ERR, "[wilc sdio]: Failed cmd52, set 0x11 data...\n");
125 /********************************************
129 ********************************************/
131 static int sdio_set_func1_block_size(u32 block_size)
139 cmd.data = (u8)block_size;
140 if (!g_sdio.sdio_cmd52(&cmd)) {
141 g_sdio.dPrint(N_ERR, "[wilc sdio]: Failed cmd52, set 0x110 data...\n");
145 cmd.data = (u8)(block_size >> 8);
146 if (!g_sdio.sdio_cmd52(&cmd)) {
147 g_sdio.dPrint(N_ERR, "[wilc sdio]: Failed cmd52, set 0x111 data...\n");
156 static int sdio_clear_int(void)
158 #ifndef WILC_SDIO_IRQ_GPIO
167 g_sdio.sdio_cmd52(&cmd);
173 if (!sdio_read_reg(WILC_HOST_RX_CTRL_0, ®)) {
174 g_sdio.dPrint(N_ERR, "[wilc spi]: Failed read reg (%08x)...\n", WILC_HOST_RX_CTRL_0);
178 sdio_write_reg(WILC_HOST_RX_CTRL_0, reg);
184 u32 sdio_xfer_cnt(void)
194 g_sdio.sdio_cmd52(&cmd);
202 g_sdio.sdio_cmd52(&cmd);
203 cnt |= (cmd.data << 8);
210 g_sdio.sdio_cmd52(&cmd);
211 cnt |= (cmd.data << 16);
216 /********************************************
220 ********************************************/
221 int sdio_check_bs(void)
233 if (!g_sdio.sdio_cmd52(&cmd)) {
234 g_sdio.dPrint(N_ERR, "[wilc sdio]: Fail cmd 52, get BS register...\n");
245 static int sdio_write_reg(u32 addr, u32 data)
248 data = BYTE_SWAP(data);
251 if ((addr >= 0xf0) && (addr <= 0xff)) {
259 if (!g_sdio.sdio_cmd52(&cmd)) {
260 g_sdio.dPrint(N_ERR, "[wilc sdio]: Failed cmd 52, read reg (%08x) ...\n", addr);
267 * set the AHB address
269 if (!sdio_set_func0_csa_address(addr))
278 cmd.buffer = (u8 *)&data;
279 cmd.block_size = g_sdio.block_size; /* johnny : prevent it from setting unexpected value */
281 if (!g_sdio.sdio_cmd53(&cmd)) {
282 g_sdio.dPrint(N_ERR, "[wilc sdio]: Failed cmd53, write reg (%08x)...\n", addr);
294 static int sdio_write(u32 addr, u8 *buf, u32 size)
296 u32 block_size = g_sdio.block_size;
303 * has to be word aligned...
317 * has to be word aligned...
331 nblk = size / block_size;
332 nleft = size % block_size;
339 cmd.block_size = block_size;
341 if (!sdio_set_func0_csa_address(addr))
344 if (!g_sdio.sdio_cmd53(&cmd)) {
345 g_sdio.dPrint(N_ERR, "[wilc sdio]: Failed cmd53 [%x], block send...\n", addr);
349 addr += nblk * block_size;
350 buf += nblk * block_size;
359 cmd.block_size = block_size; /* johnny : prevent it from setting unexpected value */
362 if (!sdio_set_func0_csa_address(addr))
365 if (!g_sdio.sdio_cmd53(&cmd)) {
366 g_sdio.dPrint(N_ERR, "[wilc sdio]: Failed cmd53 [%x], bytes send...\n", addr);
378 static int sdio_read_reg(u32 addr, u32 *data)
380 if ((addr >= 0xf0) && (addr <= 0xff)) {
387 if (!g_sdio.sdio_cmd52(&cmd)) {
388 g_sdio.dPrint(N_ERR, "[wilc sdio]: Failed cmd 52, read reg (%08x) ...\n", addr);
395 if (!sdio_set_func0_csa_address(addr))
404 cmd.buffer = (u8 *)data;
406 cmd.block_size = g_sdio.block_size; /* johnny : prevent it from setting unexpected value */
408 if (!g_sdio.sdio_cmd53(&cmd)) {
409 g_sdio.dPrint(N_ERR, "[wilc sdio]: Failed cmd53, read reg (%08x)...\n", addr);
415 *data = BYTE_SWAP(*data);
425 static int sdio_read(u32 addr, u8 *buf, u32 size)
427 u32 block_size = g_sdio.block_size;
434 * has to be word aligned...
448 * has to be word aligned...
462 nblk = size / block_size;
463 nleft = size % block_size;
470 cmd.block_size = block_size;
472 if (!sdio_set_func0_csa_address(addr))
475 if (!g_sdio.sdio_cmd53(&cmd)) {
476 g_sdio.dPrint(N_ERR, "[wilc sdio]: Failed cmd53 [%x], block read...\n", addr);
480 addr += nblk * block_size;
481 buf += nblk * block_size;
482 } /* if (nblk > 0) */
490 cmd.block_size = block_size; /* johnny : prevent it from setting unexpected value */
493 if (!sdio_set_func0_csa_address(addr))
496 if (!g_sdio.sdio_cmd53(&cmd)) {
497 g_sdio.dPrint(N_ERR, "[wilc sdio]: Failed cmd53 [%x], bytes read...\n", addr);
509 /********************************************
513 ********************************************/
515 static int sdio_deinit(void *pv)
520 static int sdio_sync(void)
525 * Disable power sequencer
527 if (!sdio_read_reg(WILC_MISC, ®)) {
528 g_sdio.dPrint(N_ERR, "[wilc sdio]: Failed read misc reg...\n");
533 if (!sdio_write_reg(WILC_MISC, reg)) {
534 g_sdio.dPrint(N_ERR, "[wilc sdio]: Failed write misc reg...\n");
538 #ifdef WILC_SDIO_IRQ_GPIO
544 * interrupt pin mux select
546 ret = sdio_read_reg(WILC_PIN_MUX_0, ®);
548 g_sdio.dPrint(N_ERR, "[wilc spi]: Failed read reg (%08x)...\n", WILC_PIN_MUX_0);
552 ret = sdio_write_reg(WILC_PIN_MUX_0, reg);
554 g_sdio.dPrint(N_ERR, "[wilc spi]: Failed write reg (%08x)...\n", WILC_PIN_MUX_0);
561 ret = sdio_read_reg(WILC_INTR_ENABLE, ®);
563 g_sdio.dPrint(N_ERR, "[wilc spi]: Failed read reg (%08x)...\n", WILC_INTR_ENABLE);
567 ret = sdio_write_reg(WILC_INTR_ENABLE, reg);
569 g_sdio.dPrint(N_ERR, "[wilc spi]: Failed write reg (%08x)...\n", WILC_INTR_ENABLE);
578 static int sdio_init(wilc_wlan_inp_t *inp, wilc_debug_func func)
584 memset(&g_sdio, 0, sizeof(wilc_sdio_t));
586 g_sdio.dPrint = func;
587 g_sdio.os_context = inp->os_context.os_private;
589 if (inp->io_func.io_init) {
590 if (!inp->io_func.io_init(g_sdio.os_context)) {
591 g_sdio.dPrint(N_ERR, "[wilc sdio]: Failed io init bus...\n");
598 g_sdio.sdio_cmd52 = inp->io_func.u.sdio.sdio_cmd52;
599 g_sdio.sdio_cmd53 = inp->io_func.u.sdio.sdio_cmd53;
600 g_sdio.sdio_set_max_speed = inp->io_func.u.sdio.sdio_set_max_speed;
601 g_sdio.sdio_set_default_speed = inp->io_func.u.sdio.sdio_set_default_speed;
604 * function 0 csa enable
611 if (!g_sdio.sdio_cmd52(&cmd)) {
612 g_sdio.dPrint(N_ERR, "[wilc sdio]: Fail cmd 52, enable csa...\n");
617 * function 0 block size
619 if (!sdio_set_func0_block_size(WILC_SDIO_BLOCK_SIZE)) {
620 g_sdio.dPrint(N_ERR, "[wilc sdio]: Fail cmd 52, set func 0 block size...\n");
623 g_sdio.block_size = WILC_SDIO_BLOCK_SIZE;
633 if (!g_sdio.sdio_cmd52(&cmd)) {
634 g_sdio.dPrint(N_ERR, "[wilc sdio] Fail cmd 52, set IOE register...\n");
639 * make sure func 1 is up
648 if (!g_sdio.sdio_cmd52(&cmd)) {
649 g_sdio.dPrint(N_ERR, "[wilc sdio]: Fail cmd 52, get IOR register...\n");
657 g_sdio.dPrint(N_ERR, "[wilc sdio]: Fail func 1 is not ready...\n");
662 * func 1 is ready, set func 1 block size
664 if (!sdio_set_func1_block_size(WILC_SDIO_BLOCK_SIZE)) {
665 g_sdio.dPrint(N_ERR, "[wilc sdio]: Fail set func 1 block size...\n");
670 * func 1 interrupt enable
677 if (!g_sdio.sdio_cmd52(&cmd)) {
678 g_sdio.dPrint(N_ERR, "[wilc sdio]: Fail cmd 52, set IEN register...\n");
683 * make sure can read back chip id correctly
685 if (!sdio_read_reg(0x1000, &chipid)) {
686 g_sdio.dPrint(N_ERR, "[wilc sdio]: Fail cmd read chip id...\n");
689 g_sdio.dPrint(N_ERR, "[wilc sdio]: chipid (%08x)\n", chipid);
690 if ((chipid & 0xfff) > 0x2a0)
691 g_sdio.has_thrpt_enh3 = 1;
693 g_sdio.has_thrpt_enh3 = 0;
694 g_sdio.dPrint(N_ERR, "[wilc sdio]: has_thrpt_enh3 = %d...\n", g_sdio.has_thrpt_enh3);
703 static void sdio_set_max_speed(void)
705 g_sdio.sdio_set_max_speed();
708 static void sdio_set_default_speed(void)
710 g_sdio.sdio_set_default_speed();
713 static int sdio_read_size(u32 *size)
720 * Read DMA count in words
727 g_sdio.sdio_cmd52(&cmd);
730 /* cmd.read_write = 0; */
731 /* cmd.function = 0; */
735 g_sdio.sdio_cmd52(&cmd);
736 tmp |= (cmd.data << 8);
742 static int sdio_read_int(u32 *int_status)
748 sdio_read_size(&tmp);
753 #ifndef WILC_SDIO_IRQ_GPIO
757 g_sdio.sdio_cmd52(&cmd);
759 if (cmd.data & (1 << 0))
761 if (cmd.data & (1 << 2))
763 if (cmd.data & (1 << 3))
765 if (cmd.data & (1 << 4))
767 if (cmd.data & (1 << 5))
769 if (cmd.data & (1 << 6))
774 for (i = g_sdio.nint; i < MAX_NUM_INT; i++) {
775 if ((tmp >> (IRG_FLAGS_OFFSET + i)) & 0x1) {
776 g_sdio.dPrint(N_ERR, "[wilc sdio]: Unexpected interrupt (1) : tmp=%x, data=%x\n", tmp, cmd.data);
790 g_sdio.sdio_cmd52(&cmd);
791 irq_flags = cmd.data & 0x1f;
792 tmp |= ((irq_flags >> 0) << IRG_FLAGS_OFFSET);
802 static int sdio_clear_int_ext(u32 val)
806 if (g_sdio.has_thrpt_enh3) {
809 #ifdef WILC_SDIO_IRQ_GPIO
813 flags = val & ((1 << MAX_NUN_INT_THRPT_ENH2) - 1);
819 /* select VMM table 0 */
820 if ((val & SEL_VMM_TBL0) == SEL_VMM_TBL0)
822 /* select VMM table 1 */
823 if ((val & SEL_VMM_TBL1) == SEL_VMM_TBL1)
826 if ((val & EN_VMM) == EN_VMM)
837 ret = g_sdio.sdio_cmd52(&cmd);
839 g_sdio.dPrint(N_ERR, "[wilc sdio]: Failed cmd52, set 0xf8 data (%d) ...\n", __LINE__);
845 #ifdef WILC_SDIO_IRQ_GPIO
847 /* see below. has_thrpt_enh2 uses register 0xf8 to clear interrupts. */
848 /* Cannot clear multiple interrupts. Must clear each interrupt individually */
851 flags = val & ((1 << MAX_NUM_INT) - 1);
856 for (i = 0; i < g_sdio.nint; i++) {
866 ret = g_sdio.sdio_cmd52(&cmd);
868 g_sdio.dPrint(N_ERR, "[wilc sdio]: Failed cmd52, set 0xf8 data (%d) ...\n", __LINE__);
879 for (i = g_sdio.nint; i < MAX_NUM_INT; i++) {
881 g_sdio.dPrint(N_ERR, "[wilc sdio]: Unexpected interrupt cleared %d...\n", i);
886 #endif /* WILC_SDIO_IRQ_GPIO */
892 /* select VMM table 0 */
893 if ((val & SEL_VMM_TBL0) == SEL_VMM_TBL0)
895 /* select VMM table 1 */
896 if ((val & SEL_VMM_TBL1) == SEL_VMM_TBL1)
899 if ((val & EN_VMM) == EN_VMM)
910 ret = g_sdio.sdio_cmd52(&cmd);
912 g_sdio.dPrint(N_ERR, "[wilc sdio]: Failed cmd52, set 0xf6 data (%d) ...\n", __LINE__);
924 static int sdio_sync_ext(int nint /* how mant interrupts to enable. */)
928 if (nint > MAX_NUM_INT) {
929 g_sdio.dPrint(N_ERR, "[wilc sdio]: Too many interupts (%d)...\n", nint);
932 if (nint > MAX_NUN_INT_THRPT_ENH2) {
933 g_sdio.dPrint(N_ERR, "[wilc sdio]: Error: Cannot support more than 5 interrupts when has_thrpt_enh2=1.\n");
940 * Disable power sequencer
942 if (!sdio_read_reg(WILC_MISC, ®)) {
943 g_sdio.dPrint(N_ERR, "[wilc sdio]: Failed read misc reg...\n");
948 if (!sdio_write_reg(WILC_MISC, reg)) {
949 g_sdio.dPrint(N_ERR, "[wilc sdio]: Failed write misc reg...\n");
953 #ifdef WILC_SDIO_IRQ_GPIO
959 * interrupt pin mux select
961 ret = sdio_read_reg(WILC_PIN_MUX_0, ®);
963 g_sdio.dPrint(N_ERR, "[wilc sdio]: Failed read reg (%08x)...\n", WILC_PIN_MUX_0);
967 ret = sdio_write_reg(WILC_PIN_MUX_0, reg);
969 g_sdio.dPrint(N_ERR, "[wilc sdio]: Failed write reg (%08x)...\n", WILC_PIN_MUX_0);
976 ret = sdio_read_reg(WILC_INTR_ENABLE, ®);
978 g_sdio.dPrint(N_ERR, "[wilc sdio]: Failed read reg (%08x)...\n", WILC_INTR_ENABLE);
982 for (i = 0; (i < 5) && (nint > 0); i++, nint--)
983 reg |= (1 << (27 + i));
984 ret = sdio_write_reg(WILC_INTR_ENABLE, reg);
986 g_sdio.dPrint(N_ERR, "[wilc sdio]: Failed write reg (%08x)...\n", WILC_INTR_ENABLE);
990 ret = sdio_read_reg(WILC_INTR2_ENABLE, ®);
992 g_sdio.dPrint(N_ERR, "[wilc sdio]: Failed read reg (%08x)...\n", WILC_INTR2_ENABLE);
996 for (i = 0; (i < 3) && (nint > 0); i++, nint--)
999 ret = sdio_read_reg(WILC_INTR2_ENABLE, ®);
1001 g_sdio.dPrint(N_ERR, "[wilc sdio]: Failed write reg (%08x)...\n", WILC_INTR2_ENABLE);
1006 #endif /* WILC_SDIO_IRQ_GPIO */
1010 /********************************************
1012 * Global sdio HIF function table
1014 ********************************************/
1016 wilc_hif_func_t hif_sdio = {
1033 sdio_set_default_speed,