diff options
Diffstat (limited to 'drivers/mmc')
-rw-r--r-- | drivers/mmc/Kconfig | 2 | ||||
-rw-r--r-- | drivers/mmc/at91_mci.c | 3 | ||||
-rw-r--r-- | drivers/mmc/au1xmmc.c | 13 | ||||
-rw-r--r-- | drivers/mmc/imxmmc.c | 4 | ||||
-rw-r--r-- | drivers/mmc/mmc.c | 182 | ||||
-rw-r--r-- | drivers/mmc/mmc_block.c | 16 | ||||
-rw-r--r-- | drivers/mmc/mmc_queue.c | 2 | ||||
-rw-r--r-- | drivers/mmc/mmc_sysfs.c | 2 | ||||
-rw-r--r-- | drivers/mmc/mmci.c | 15 | ||||
-rw-r--r-- | drivers/mmc/omap.c | 6 | ||||
-rw-r--r-- | drivers/mmc/pxamci.c | 10 | ||||
-rw-r--r-- | drivers/mmc/sdhci.c | 131 | ||||
-rw-r--r-- | drivers/mmc/sdhci.h | 2 | ||||
-rw-r--r-- | drivers/mmc/tifm_sd.c | 494 | ||||
-rw-r--r-- | drivers/mmc/wbsd.c | 108 | ||||
-rw-r--r-- | drivers/mmc/wbsd.h | 1 |
16 files changed, 601 insertions, 390 deletions
diff --git a/drivers/mmc/Kconfig b/drivers/mmc/Kconfig index 4224686fdf2a..12af9c718764 100644 --- a/drivers/mmc/Kconfig +++ b/drivers/mmc/Kconfig | |||
@@ -111,7 +111,7 @@ config MMC_IMX | |||
111 | 111 | ||
112 | config MMC_TIFM_SD | 112 | config MMC_TIFM_SD |
113 | tristate "TI Flash Media MMC/SD Interface support (EXPERIMENTAL)" | 113 | tristate "TI Flash Media MMC/SD Interface support (EXPERIMENTAL)" |
114 | depends on MMC && EXPERIMENTAL | 114 | depends on MMC && EXPERIMENTAL && PCI |
115 | select TIFM_CORE | 115 | select TIFM_CORE |
116 | help | 116 | help |
117 | Say Y here if you want to be able to access MMC/SD cards with | 117 | Say Y here if you want to be able to access MMC/SD cards with |
diff --git a/drivers/mmc/at91_mci.c b/drivers/mmc/at91_mci.c index 521ace9a4db0..459f4b4feded 100644 --- a/drivers/mmc/at91_mci.c +++ b/drivers/mmc/at91_mci.c | |||
@@ -823,6 +823,9 @@ static int __init at91_mci_probe(struct platform_device *pdev) | |||
823 | mmc->ocr_avail = MMC_VDD_32_33 | MMC_VDD_33_34; | 823 | mmc->ocr_avail = MMC_VDD_32_33 | MMC_VDD_33_34; |
824 | mmc->caps = MMC_CAP_BYTEBLOCK; | 824 | mmc->caps = MMC_CAP_BYTEBLOCK; |
825 | 825 | ||
826 | mmc->max_blk_size = 4095; | ||
827 | mmc->max_blk_count = mmc->max_req_size; | ||
828 | |||
826 | host = mmc_priv(mmc); | 829 | host = mmc_priv(mmc); |
827 | host->mmc = mmc; | 830 | host->mmc = mmc; |
828 | host->buffer = NULL; | 831 | host->buffer = NULL; |
diff --git a/drivers/mmc/au1xmmc.c b/drivers/mmc/au1xmmc.c index 800527cf40d5..b834be261ab7 100644 --- a/drivers/mmc/au1xmmc.c +++ b/drivers/mmc/au1xmmc.c | |||
@@ -152,8 +152,9 @@ static inline int au1xmmc_card_inserted(struct au1xmmc_host *host) | |||
152 | ? 1 : 0; | 152 | ? 1 : 0; |
153 | } | 153 | } |
154 | 154 | ||
155 | static inline int au1xmmc_card_readonly(struct au1xmmc_host *host) | 155 | static int au1xmmc_card_readonly(struct mmc_host *mmc) |
156 | { | 156 | { |
157 | struct au1xmmc_host *host = mmc_priv(mmc); | ||
157 | return (bcsr->status & au1xmmc_card_table[host->id].wpstatus) | 158 | return (bcsr->status & au1xmmc_card_table[host->id].wpstatus) |
158 | ? 1 : 0; | 159 | ? 1 : 0; |
159 | } | 160 | } |
@@ -193,6 +194,8 @@ static int au1xmmc_send_command(struct au1xmmc_host *host, int wait, | |||
193 | u32 mmccmd = (cmd->opcode << SD_CMD_CI_SHIFT); | 194 | u32 mmccmd = (cmd->opcode << SD_CMD_CI_SHIFT); |
194 | 195 | ||
195 | switch (mmc_resp_type(cmd)) { | 196 | switch (mmc_resp_type(cmd)) { |
197 | case MMC_RSP_NONE: | ||
198 | break; | ||
196 | case MMC_RSP_R1: | 199 | case MMC_RSP_R1: |
197 | mmccmd |= SD_CMD_RT_1; | 200 | mmccmd |= SD_CMD_RT_1; |
198 | break; | 201 | break; |
@@ -205,6 +208,10 @@ static int au1xmmc_send_command(struct au1xmmc_host *host, int wait, | |||
205 | case MMC_RSP_R3: | 208 | case MMC_RSP_R3: |
206 | mmccmd |= SD_CMD_RT_3; | 209 | mmccmd |= SD_CMD_RT_3; |
207 | break; | 210 | break; |
211 | default: | ||
212 | printk(KERN_INFO "au1xmmc: unhandled response type %02x\n", | ||
213 | mmc_resp_type(cmd)); | ||
214 | return MMC_ERR_INVALID; | ||
208 | } | 215 | } |
209 | 216 | ||
210 | switch(cmd->opcode) { | 217 | switch(cmd->opcode) { |
@@ -878,6 +885,7 @@ static void au1xmmc_init_dma(struct au1xmmc_host *host) | |||
878 | static const struct mmc_host_ops au1xmmc_ops = { | 885 | static const struct mmc_host_ops au1xmmc_ops = { |
879 | .request = au1xmmc_request, | 886 | .request = au1xmmc_request, |
880 | .set_ios = au1xmmc_set_ios, | 887 | .set_ios = au1xmmc_set_ios, |
888 | .get_ro = au1xmmc_card_readonly, | ||
881 | }; | 889 | }; |
882 | 890 | ||
883 | static int __devinit au1xmmc_probe(struct platform_device *pdev) | 891 | static int __devinit au1xmmc_probe(struct platform_device *pdev) |
@@ -914,6 +922,9 @@ static int __devinit au1xmmc_probe(struct platform_device *pdev) | |||
914 | mmc->max_seg_size = AU1XMMC_DESCRIPTOR_SIZE; | 922 | mmc->max_seg_size = AU1XMMC_DESCRIPTOR_SIZE; |
915 | mmc->max_phys_segs = AU1XMMC_DESCRIPTOR_COUNT; | 923 | mmc->max_phys_segs = AU1XMMC_DESCRIPTOR_COUNT; |
916 | 924 | ||
925 | mmc->max_blk_size = 2048; | ||
926 | mmc->max_blk_count = 512; | ||
927 | |||
917 | mmc->ocr_avail = AU1XMMC_OCR; | 928 | mmc->ocr_avail = AU1XMMC_OCR; |
918 | 929 | ||
919 | host = mmc_priv(mmc); | 930 | host = mmc_priv(mmc); |
diff --git a/drivers/mmc/imxmmc.c b/drivers/mmc/imxmmc.c index bfb9ff693208..b060d4bfba29 100644 --- a/drivers/mmc/imxmmc.c +++ b/drivers/mmc/imxmmc.c | |||
@@ -958,8 +958,10 @@ static int imxmci_probe(struct platform_device *pdev) | |||
958 | /* MMC core transfer sizes tunable parameters */ | 958 | /* MMC core transfer sizes tunable parameters */ |
959 | mmc->max_hw_segs = 64; | 959 | mmc->max_hw_segs = 64; |
960 | mmc->max_phys_segs = 64; | 960 | mmc->max_phys_segs = 64; |
961 | mmc->max_sectors = 64; /* default 1 << (PAGE_CACHE_SHIFT - 9) */ | ||
962 | mmc->max_seg_size = 64*512; /* default PAGE_CACHE_SIZE */ | 961 | mmc->max_seg_size = 64*512; /* default PAGE_CACHE_SIZE */ |
962 | mmc->max_req_size = 64*512; /* default PAGE_CACHE_SIZE */ | ||
963 | mmc->max_blk_size = 2048; | ||
964 | mmc->max_blk_count = 65535; | ||
963 | 965 | ||
964 | host = mmc_priv(mmc); | 966 | host = mmc_priv(mmc); |
965 | host->mmc = mmc; | 967 | host->mmc = mmc; |
diff --git a/drivers/mmc/mmc.c b/drivers/mmc/mmc.c index 6f2a282e2b97..5046a1661342 100644 --- a/drivers/mmc/mmc.c +++ b/drivers/mmc/mmc.c | |||
@@ -103,11 +103,16 @@ mmc_start_request(struct mmc_host *host, struct mmc_request *mrq) | |||
103 | mmc_hostname(host), mrq->cmd->opcode, | 103 | mmc_hostname(host), mrq->cmd->opcode, |
104 | mrq->cmd->arg, mrq->cmd->flags); | 104 | mrq->cmd->arg, mrq->cmd->flags); |
105 | 105 | ||
106 | WARN_ON(host->card_busy == NULL); | 106 | WARN_ON(!host->claimed); |
107 | 107 | ||
108 | mrq->cmd->error = 0; | 108 | mrq->cmd->error = 0; |
109 | mrq->cmd->mrq = mrq; | 109 | mrq->cmd->mrq = mrq; |
110 | if (mrq->data) { | 110 | if (mrq->data) { |
111 | BUG_ON(mrq->data->blksz > host->max_blk_size); | ||
112 | BUG_ON(mrq->data->blocks > host->max_blk_count); | ||
113 | BUG_ON(mrq->data->blocks * mrq->data->blksz > | ||
114 | host->max_req_size); | ||
115 | |||
111 | mrq->cmd->data = mrq->data; | 116 | mrq->cmd->data = mrq->data; |
112 | mrq->data->error = 0; | 117 | mrq->data->error = 0; |
113 | mrq->data->mrq = mrq; | 118 | mrq->data->mrq = mrq; |
@@ -157,7 +162,7 @@ int mmc_wait_for_cmd(struct mmc_host *host, struct mmc_command *cmd, int retries | |||
157 | { | 162 | { |
158 | struct mmc_request mrq; | 163 | struct mmc_request mrq; |
159 | 164 | ||
160 | BUG_ON(host->card_busy == NULL); | 165 | BUG_ON(!host->claimed); |
161 | 166 | ||
162 | memset(&mrq, 0, sizeof(struct mmc_request)); | 167 | memset(&mrq, 0, sizeof(struct mmc_request)); |
163 | 168 | ||
@@ -195,7 +200,7 @@ int mmc_wait_for_app_cmd(struct mmc_host *host, unsigned int rca, | |||
195 | 200 | ||
196 | int i, err; | 201 | int i, err; |
197 | 202 | ||
198 | BUG_ON(host->card_busy == NULL); | 203 | BUG_ON(!host->claimed); |
199 | BUG_ON(retries < 0); | 204 | BUG_ON(retries < 0); |
200 | 205 | ||
201 | err = MMC_ERR_INVALID; | 206 | err = MMC_ERR_INVALID; |
@@ -289,7 +294,10 @@ void mmc_set_data_timeout(struct mmc_data *data, const struct mmc_card *card, | |||
289 | else | 294 | else |
290 | limit_us = 100000; | 295 | limit_us = 100000; |
291 | 296 | ||
292 | if (timeout_us > limit_us) { | 297 | /* |
298 | * SDHC cards always use these fixed values. | ||
299 | */ | ||
300 | if (timeout_us > limit_us || mmc_card_blockaddr(card)) { | ||
293 | data->timeout_ns = limit_us * 1000; | 301 | data->timeout_ns = limit_us * 1000; |
294 | data->timeout_clks = 0; | 302 | data->timeout_clks = 0; |
295 | } | 303 | } |
@@ -320,14 +328,14 @@ int __mmc_claim_host(struct mmc_host *host, struct mmc_card *card) | |||
320 | spin_lock_irqsave(&host->lock, flags); | 328 | spin_lock_irqsave(&host->lock, flags); |
321 | while (1) { | 329 | while (1) { |
322 | set_current_state(TASK_UNINTERRUPTIBLE); | 330 | set_current_state(TASK_UNINTERRUPTIBLE); |
323 | if (host->card_busy == NULL) | 331 | if (!host->claimed) |
324 | break; | 332 | break; |
325 | spin_unlock_irqrestore(&host->lock, flags); | 333 | spin_unlock_irqrestore(&host->lock, flags); |
326 | schedule(); | 334 | schedule(); |
327 | spin_lock_irqsave(&host->lock, flags); | 335 | spin_lock_irqsave(&host->lock, flags); |
328 | } | 336 | } |
329 | set_current_state(TASK_RUNNING); | 337 | set_current_state(TASK_RUNNING); |
330 | host->card_busy = card; | 338 | host->claimed = 1; |
331 | spin_unlock_irqrestore(&host->lock, flags); | 339 | spin_unlock_irqrestore(&host->lock, flags); |
332 | remove_wait_queue(&host->wq, &wait); | 340 | remove_wait_queue(&host->wq, &wait); |
333 | 341 | ||
@@ -353,10 +361,10 @@ void mmc_release_host(struct mmc_host *host) | |||
353 | { | 361 | { |
354 | unsigned long flags; | 362 | unsigned long flags; |
355 | 363 | ||
356 | BUG_ON(host->card_busy == NULL); | 364 | BUG_ON(!host->claimed); |
357 | 365 | ||
358 | spin_lock_irqsave(&host->lock, flags); | 366 | spin_lock_irqsave(&host->lock, flags); |
359 | host->card_busy = NULL; | 367 | host->claimed = 0; |
360 | spin_unlock_irqrestore(&host->lock, flags); | 368 | spin_unlock_irqrestore(&host->lock, flags); |
361 | 369 | ||
362 | wake_up(&host->wq); | 370 | wake_up(&host->wq); |
@@ -372,7 +380,7 @@ static inline void mmc_set_ios(struct mmc_host *host) | |||
372 | mmc_hostname(host), ios->clock, ios->bus_mode, | 380 | mmc_hostname(host), ios->clock, ios->bus_mode, |
373 | ios->power_mode, ios->chip_select, ios->vdd, | 381 | ios->power_mode, ios->chip_select, ios->vdd, |
374 | ios->bus_width); | 382 | ios->bus_width); |
375 | 383 | ||
376 | host->ops->set_ios(host, ios); | 384 | host->ops->set_ios(host, ios); |
377 | } | 385 | } |
378 | 386 | ||
@@ -381,7 +389,7 @@ static int mmc_select_card(struct mmc_host *host, struct mmc_card *card) | |||
381 | int err; | 389 | int err; |
382 | struct mmc_command cmd; | 390 | struct mmc_command cmd; |
383 | 391 | ||
384 | BUG_ON(host->card_busy == NULL); | 392 | BUG_ON(!host->claimed); |
385 | 393 | ||
386 | if (host->card_selected == card) | 394 | if (host->card_selected == card) |
387 | return MMC_ERR_NONE; | 395 | return MMC_ERR_NONE; |
@@ -588,34 +596,65 @@ static void mmc_decode_csd(struct mmc_card *card) | |||
588 | 596 | ||
589 | if (mmc_card_sd(card)) { | 597 | if (mmc_card_sd(card)) { |
590 | csd_struct = UNSTUFF_BITS(resp, 126, 2); | 598 | csd_struct = UNSTUFF_BITS(resp, 126, 2); |
591 | if (csd_struct != 0) { | 599 | |
600 | switch (csd_struct) { | ||
601 | case 0: | ||
602 | m = UNSTUFF_BITS(resp, 115, 4); | ||
603 | e = UNSTUFF_BITS(resp, 112, 3); | ||
604 | csd->tacc_ns = (tacc_exp[e] * tacc_mant[m] + 9) / 10; | ||
605 | csd->tacc_clks = UNSTUFF_BITS(resp, 104, 8) * 100; | ||
606 | |||
607 | m = UNSTUFF_BITS(resp, 99, 4); | ||
608 | e = UNSTUFF_BITS(resp, 96, 3); | ||
609 | csd->max_dtr = tran_exp[e] * tran_mant[m]; | ||
610 | csd->cmdclass = UNSTUFF_BITS(resp, 84, 12); | ||
611 | |||
612 | e = UNSTUFF_BITS(resp, 47, 3); | ||
613 | m = UNSTUFF_BITS(resp, 62, 12); | ||
614 | csd->capacity = (1 + m) << (e + 2); | ||
615 | |||
616 | csd->read_blkbits = UNSTUFF_BITS(resp, 80, 4); | ||
617 | csd->read_partial = UNSTUFF_BITS(resp, 79, 1); | ||
618 | csd->write_misalign = UNSTUFF_BITS(resp, 78, 1); | ||
619 | csd->read_misalign = UNSTUFF_BITS(resp, 77, 1); | ||
620 | csd->r2w_factor = UNSTUFF_BITS(resp, 26, 3); | ||
621 | csd->write_blkbits = UNSTUFF_BITS(resp, 22, 4); | ||
622 | csd->write_partial = UNSTUFF_BITS(resp, 21, 1); | ||
623 | break; | ||
624 | case 1: | ||
625 | /* | ||
626 | * This is a block-addressed SDHC card. Most | ||
627 | * interesting fields are unused and have fixed | ||
628 | * values. To avoid getting tripped by buggy cards, | ||
629 | * we assume those fixed values ourselves. | ||
630 | */ | ||
631 | mmc_card_set_blockaddr(card); | ||
632 | |||
633 | csd->tacc_ns = 0; /* Unused */ | ||
634 | csd->tacc_clks = 0; /* Unused */ | ||
635 | |||
636 | m = UNSTUFF_BITS(resp, 99, 4); | ||
637 | e = UNSTUFF_BITS(resp, 96, 3); | ||
638 | csd->max_dtr = tran_exp[e] * tran_mant[m]; | ||
639 | csd->cmdclass = UNSTUFF_BITS(resp, 84, 12); | ||
640 | |||
641 | m = UNSTUFF_BITS(resp, 48, 22); | ||
642 | csd->capacity = (1 + m) << 10; | ||
643 | |||
644 | csd->read_blkbits = 9; | ||
645 | csd->read_partial = 0; | ||
646 | csd->write_misalign = 0; | ||
647 | csd->read_misalign = 0; | ||
648 | csd->r2w_factor = 4; /* Unused */ | ||
649 | csd->write_blkbits = 9; | ||
650 | csd->write_partial = 0; | ||
651 | break; | ||
652 | default: | ||
592 | printk("%s: unrecognised CSD structure version %d\n", | 653 | printk("%s: unrecognised CSD structure version %d\n", |
593 | mmc_hostname(card->host), csd_struct); | 654 | mmc_hostname(card->host), csd_struct); |
594 | mmc_card_set_bad(card); | 655 | mmc_card_set_bad(card); |
595 | return; | 656 | return; |
596 | } | 657 | } |
597 | |||
598 | m = UNSTUFF_BITS(resp, 115, 4); | ||
599 | e = UNSTUFF_BITS(resp, 112, 3); | ||
600 | csd->tacc_ns = (tacc_exp[e] * tacc_mant[m] + 9) / 10; | ||
601 | csd->tacc_clks = UNSTUFF_BITS(resp, 104, 8) * 100; | ||
602 | |||
603 | m = UNSTUFF_BITS(resp, 99, 4); | ||
604 | e = UNSTUFF_BITS(resp, 96, 3); | ||
605 | csd->max_dtr = tran_exp[e] * tran_mant[m]; | ||
606 | csd->cmdclass = UNSTUFF_BITS(resp, 84, 12); | ||
607 | |||
608 | e = UNSTUFF_BITS(resp, 47, 3); | ||
609 | m = UNSTUFF_BITS(resp, 62, 12); | ||
610 | csd->capacity = (1 + m) << (e + 2); | ||
611 | |||
612 | csd->read_blkbits = UNSTUFF_BITS(resp, 80, 4); | ||
613 | csd->read_partial = UNSTUFF_BITS(resp, 79, 1); | ||
614 | csd->write_misalign = UNSTUFF_BITS(resp, 78, 1); | ||
615 | csd->read_misalign = UNSTUFF_BITS(resp, 77, 1); | ||
616 | csd->r2w_factor = UNSTUFF_BITS(resp, 26, 3); | ||
617 | csd->write_blkbits = UNSTUFF_BITS(resp, 22, 4); | ||
618 | csd->write_partial = UNSTUFF_BITS(resp, 21, 1); | ||
619 | } else { | 658 | } else { |
620 | /* | 659 | /* |
621 | * We only understand CSD structure v1.1 and v1.2. | 660 | * We only understand CSD structure v1.1 and v1.2. |
@@ -848,6 +887,41 @@ static int mmc_send_app_op_cond(struct mmc_host *host, u32 ocr, u32 *rocr) | |||
848 | return err; | 887 | return err; |
849 | } | 888 | } |
850 | 889 | ||
890 | static int mmc_send_if_cond(struct mmc_host *host, u32 ocr, int *rsd2) | ||
891 | { | ||
892 | struct mmc_command cmd; | ||
893 | int err, sd2; | ||
894 | static const u8 test_pattern = 0xAA; | ||
895 | |||
896 | /* | ||
897 | * To support SD 2.0 cards, we must always invoke SD_SEND_IF_COND | ||
898 | * before SD_APP_OP_COND. This command will harmlessly fail for | ||
899 | * SD 1.0 cards. | ||
900 | */ | ||
901 | cmd.opcode = SD_SEND_IF_COND; | ||
902 | cmd.arg = ((ocr & 0xFF8000) != 0) << 8 | test_pattern; | ||
903 | cmd.flags = MMC_RSP_R7 | MMC_CMD_BCR; | ||
904 | |||
905 | err = mmc_wait_for_cmd(host, &cmd, 0); | ||
906 | if (err == MMC_ERR_NONE) { | ||
907 | if ((cmd.resp[0] & 0xFF) == test_pattern) { | ||
908 | sd2 = 1; | ||
909 | } else { | ||
910 | sd2 = 0; | ||
911 | err = MMC_ERR_FAILED; | ||
912 | } | ||
913 | } else { | ||
914 | /* | ||
915 | * Treat errors as SD 1.0 card. | ||
916 | */ | ||
917 | sd2 = 0; | ||
918 | err = MMC_ERR_NONE; | ||
919 | } | ||
920 | if (rsd2) | ||
921 | *rsd2 = sd2; | ||
922 | return err; | ||
923 | } | ||
924 | |||
851 | /* | 925 | /* |
852 | * Discover cards by requesting their CID. If this command | 926 | * Discover cards by requesting their CID. If this command |
853 | * times out, it is not an error; there are no further cards | 927 | * times out, it is not an error; there are no further cards |
@@ -1018,7 +1092,8 @@ static void mmc_process_ext_csds(struct mmc_host *host) | |||
1018 | mmc_wait_for_req(host, &mrq); | 1092 | mmc_wait_for_req(host, &mrq); |
1019 | 1093 | ||
1020 | if (cmd.error != MMC_ERR_NONE || data.error != MMC_ERR_NONE) { | 1094 | if (cmd.error != MMC_ERR_NONE || data.error != MMC_ERR_NONE) { |
1021 | mmc_card_set_dead(card); | 1095 | printk("%s: unable to read EXT_CSD, performance " |
1096 | "might suffer.\n", mmc_hostname(card->host)); | ||
1022 | continue; | 1097 | continue; |
1023 | } | 1098 | } |
1024 | 1099 | ||
@@ -1034,7 +1109,6 @@ static void mmc_process_ext_csds(struct mmc_host *host) | |||
1034 | printk("%s: card is mmc v4 but doesn't support " | 1109 | printk("%s: card is mmc v4 but doesn't support " |
1035 | "any high-speed modes.\n", | 1110 | "any high-speed modes.\n", |
1036 | mmc_hostname(card->host)); | 1111 | mmc_hostname(card->host)); |
1037 | mmc_card_set_bad(card); | ||
1038 | continue; | 1112 | continue; |
1039 | } | 1113 | } |
1040 | 1114 | ||
@@ -1215,7 +1289,9 @@ static void mmc_read_switch_caps(struct mmc_host *host) | |||
1215 | mmc_wait_for_req(host, &mrq); | 1289 | mmc_wait_for_req(host, &mrq); |
1216 | 1290 | ||
1217 | if (cmd.error != MMC_ERR_NONE || data.error != MMC_ERR_NONE) { | 1291 | if (cmd.error != MMC_ERR_NONE || data.error != MMC_ERR_NONE) { |
1218 | mmc_card_set_dead(card); | 1292 | printk("%s: unable to read switch capabilities, " |
1293 | "performance might suffer.\n", | ||
1294 | mmc_hostname(card->host)); | ||
1219 | continue; | 1295 | continue; |
1220 | } | 1296 | } |
1221 | 1297 | ||
@@ -1247,12 +1323,8 @@ static void mmc_read_switch_caps(struct mmc_host *host) | |||
1247 | 1323 | ||
1248 | mmc_wait_for_req(host, &mrq); | 1324 | mmc_wait_for_req(host, &mrq); |
1249 | 1325 | ||
1250 | if (cmd.error != MMC_ERR_NONE || data.error != MMC_ERR_NONE) { | 1326 | if (cmd.error != MMC_ERR_NONE || data.error != MMC_ERR_NONE || |
1251 | mmc_card_set_dead(card); | 1327 | (status[16] & 0xF) != 1) { |
1252 | continue; | ||
1253 | } | ||
1254 | |||
1255 | if ((status[16] & 0xF) != 1) { | ||
1256 | printk(KERN_WARNING "%s: Problem switching card " | 1328 | printk(KERN_WARNING "%s: Problem switching card " |
1257 | "into high-speed mode!\n", | 1329 | "into high-speed mode!\n", |
1258 | mmc_hostname(host)); | 1330 | mmc_hostname(host)); |
@@ -1334,6 +1406,10 @@ static void mmc_setup(struct mmc_host *host) | |||
1334 | mmc_power_up(host); | 1406 | mmc_power_up(host); |
1335 | mmc_idle_cards(host); | 1407 | mmc_idle_cards(host); |
1336 | 1408 | ||
1409 | err = mmc_send_if_cond(host, host->ocr_avail, NULL); | ||
1410 | if (err != MMC_ERR_NONE) { | ||
1411 | return; | ||
1412 | } | ||
1337 | err = mmc_send_app_op_cond(host, 0, &ocr); | 1413 | err = mmc_send_app_op_cond(host, 0, &ocr); |
1338 | 1414 | ||
1339 | /* | 1415 | /* |
@@ -1386,10 +1462,21 @@ static void mmc_setup(struct mmc_host *host) | |||
1386 | * all get the idea that they should be ready for CMD2. | 1462 | * all get the idea that they should be ready for CMD2. |
1387 | * (My SanDisk card seems to need this.) | 1463 | * (My SanDisk card seems to need this.) |
1388 | */ | 1464 | */ |
1389 | if (host->mode == MMC_MODE_SD) | 1465 | if (host->mode == MMC_MODE_SD) { |
1390 | mmc_send_app_op_cond(host, host->ocr, NULL); | 1466 | int err, sd2; |
1391 | else | 1467 | err = mmc_send_if_cond(host, host->ocr, &sd2); |
1468 | if (err == MMC_ERR_NONE) { | ||
1469 | /* | ||
1470 | * If SD_SEND_IF_COND indicates an SD 2.0 | ||
1471 | * compliant card and we should set bit 30 | ||
1472 | * of the ocr to indicate that we can handle | ||
1473 | * block-addressed SDHC cards. | ||
1474 | */ | ||
1475 | mmc_send_app_op_cond(host, host->ocr | (sd2 << 30), NULL); | ||
1476 | } | ||
1477 | } else { | ||
1392 | mmc_send_op_cond(host, host->ocr, NULL); | 1478 | mmc_send_op_cond(host, host->ocr, NULL); |
1479 | } | ||
1393 | 1480 | ||
1394 | mmc_discover_cards(host); | 1481 | mmc_discover_cards(host); |
1395 | 1482 | ||
@@ -1519,8 +1606,11 @@ struct mmc_host *mmc_alloc_host(int extra, struct device *dev) | |||
1519 | */ | 1606 | */ |
1520 | host->max_hw_segs = 1; | 1607 | host->max_hw_segs = 1; |
1521 | host->max_phys_segs = 1; | 1608 | host->max_phys_segs = 1; |
1522 | host->max_sectors = 1 << (PAGE_CACHE_SHIFT - 9); | ||
1523 | host->max_seg_size = PAGE_CACHE_SIZE; | 1609 | host->max_seg_size = PAGE_CACHE_SIZE; |
1610 | |||
1611 | host->max_req_size = PAGE_CACHE_SIZE; | ||
1612 | host->max_blk_size = 512; | ||
1613 | host->max_blk_count = PAGE_CACHE_SIZE / 512; | ||
1524 | } | 1614 | } |
1525 | 1615 | ||
1526 | return host; | 1616 | return host; |
diff --git a/drivers/mmc/mmc_block.c b/drivers/mmc/mmc_block.c index 87713572293f..86439a0bb271 100644 --- a/drivers/mmc/mmc_block.c +++ b/drivers/mmc/mmc_block.c | |||
@@ -20,7 +20,6 @@ | |||
20 | #include <linux/module.h> | 20 | #include <linux/module.h> |
21 | #include <linux/init.h> | 21 | #include <linux/init.h> |
22 | 22 | ||
23 | #include <linux/sched.h> | ||
24 | #include <linux/kernel.h> | 23 | #include <linux/kernel.h> |
25 | #include <linux/fs.h> | 24 | #include <linux/fs.h> |
26 | #include <linux/errno.h> | 25 | #include <linux/errno.h> |
@@ -237,13 +236,17 @@ static int mmc_blk_issue_rq(struct mmc_queue *mq, struct request *req) | |||
237 | brq.mrq.cmd = &brq.cmd; | 236 | brq.mrq.cmd = &brq.cmd; |
238 | brq.mrq.data = &brq.data; | 237 | brq.mrq.data = &brq.data; |
239 | 238 | ||
240 | brq.cmd.arg = req->sector << 9; | 239 | brq.cmd.arg = req->sector; |
240 | if (!mmc_card_blockaddr(card)) | ||
241 | brq.cmd.arg <<= 9; | ||
241 | brq.cmd.flags = MMC_RSP_R1 | MMC_CMD_ADTC; | 242 | brq.cmd.flags = MMC_RSP_R1 | MMC_CMD_ADTC; |
242 | brq.data.blksz = 1 << md->block_bits; | 243 | brq.data.blksz = 1 << md->block_bits; |
243 | brq.data.blocks = req->nr_sectors >> (md->block_bits - 9); | ||
244 | brq.stop.opcode = MMC_STOP_TRANSMISSION; | 244 | brq.stop.opcode = MMC_STOP_TRANSMISSION; |
245 | brq.stop.arg = 0; | 245 | brq.stop.arg = 0; |
246 | brq.stop.flags = MMC_RSP_R1B | MMC_CMD_AC; | 246 | brq.stop.flags = MMC_RSP_R1B | MMC_CMD_AC; |
247 | brq.data.blocks = req->nr_sectors >> (md->block_bits - 9); | ||
248 | if (brq.data.blocks > card->host->max_blk_count) | ||
249 | brq.data.blocks = card->host->max_blk_count; | ||
247 | 250 | ||
248 | mmc_set_data_timeout(&brq.data, card, rq_data_dir(req) != READ); | 251 | mmc_set_data_timeout(&brq.data, card, rq_data_dir(req) != READ); |
249 | 252 | ||
@@ -375,9 +378,10 @@ static int mmc_blk_issue_rq(struct mmc_queue *mq, struct request *req) | |||
375 | spin_unlock_irq(&md->lock); | 378 | spin_unlock_irq(&md->lock); |
376 | } | 379 | } |
377 | 380 | ||
381 | flush_queue: | ||
382 | |||
378 | mmc_card_release_host(card); | 383 | mmc_card_release_host(card); |
379 | 384 | ||
380 | flush_queue: | ||
381 | spin_lock_irq(&md->lock); | 385 | spin_lock_irq(&md->lock); |
382 | while (ret) { | 386 | while (ret) { |
383 | ret = end_that_request_chunk(req, 0, | 387 | ret = end_that_request_chunk(req, 0, |
@@ -494,6 +498,10 @@ mmc_blk_set_blksize(struct mmc_blk_data *md, struct mmc_card *card) | |||
494 | struct mmc_command cmd; | 498 | struct mmc_command cmd; |
495 | int err; | 499 | int err; |
496 | 500 | ||
501 | /* Block-addressed cards ignore MMC_SET_BLOCKLEN. */ | ||
502 | if (mmc_card_blockaddr(card)) | ||
503 | return 0; | ||
504 | |||
497 | mmc_card_claim_host(card); | 505 | mmc_card_claim_host(card); |
498 | cmd.opcode = MMC_SET_BLOCKLEN; | 506 | cmd.opcode = MMC_SET_BLOCKLEN; |
499 | cmd.arg = 1 << md->block_bits; | 507 | cmd.arg = 1 << md->block_bits; |
diff --git a/drivers/mmc/mmc_queue.c b/drivers/mmc/mmc_queue.c index 3e35a43819fb..c27e42645cdb 100644 --- a/drivers/mmc/mmc_queue.c +++ b/drivers/mmc/mmc_queue.c | |||
@@ -147,7 +147,7 @@ int mmc_init_queue(struct mmc_queue *mq, struct mmc_card *card, spinlock_t *lock | |||
147 | 147 | ||
148 | blk_queue_prep_rq(mq->queue, mmc_prep_request); | 148 | blk_queue_prep_rq(mq->queue, mmc_prep_request); |
149 | blk_queue_bounce_limit(mq->queue, limit); | 149 | blk_queue_bounce_limit(mq->queue, limit); |
150 | blk_queue_max_sectors(mq->queue, host->max_sectors); | 150 | blk_queue_max_sectors(mq->queue, host->max_req_size / 512); |
151 | blk_queue_max_phys_segments(mq->queue, host->max_phys_segs); | 151 | blk_queue_max_phys_segments(mq->queue, host->max_phys_segs); |
152 | blk_queue_max_hw_segments(mq->queue, host->max_hw_segs); | 152 | blk_queue_max_hw_segments(mq->queue, host->max_hw_segs); |
153 | blk_queue_max_segment_size(mq->queue, host->max_seg_size); | 153 | blk_queue_max_segment_size(mq->queue, host->max_seg_size); |
diff --git a/drivers/mmc/mmc_sysfs.c b/drivers/mmc/mmc_sysfs.c index e334acd045bc..d32698b02d7f 100644 --- a/drivers/mmc/mmc_sysfs.c +++ b/drivers/mmc/mmc_sysfs.c | |||
@@ -199,7 +199,7 @@ void mmc_init_card(struct mmc_card *card, struct mmc_host *host) | |||
199 | memset(card, 0, sizeof(struct mmc_card)); | 199 | memset(card, 0, sizeof(struct mmc_card)); |
200 | card->host = host; | 200 | card->host = host; |
201 | device_initialize(&card->dev); | 201 | device_initialize(&card->dev); |
202 | card->dev.parent = mmc_dev(host); | 202 | card->dev.parent = mmc_classdev(host); |
203 | card->dev.bus = &mmc_bus_type; | 203 | card->dev.bus = &mmc_bus_type; |
204 | card->dev.release = mmc_release_card; | 204 | card->dev.release = mmc_release_card; |
205 | } | 205 | } |
diff --git a/drivers/mmc/mmci.c b/drivers/mmc/mmci.c index ccfe6561be24..5941dd951e82 100644 --- a/drivers/mmc/mmci.c +++ b/drivers/mmc/mmci.c | |||
@@ -524,15 +524,24 @@ static int mmci_probe(struct amba_device *dev, void *id) | |||
524 | /* | 524 | /* |
525 | * Since we only have a 16-bit data length register, we must | 525 | * Since we only have a 16-bit data length register, we must |
526 | * ensure that we don't exceed 2^16-1 bytes in a single request. | 526 | * ensure that we don't exceed 2^16-1 bytes in a single request. |
527 | * Choose 64 (512-byte) sectors as the limit. | ||
528 | */ | 527 | */ |
529 | mmc->max_sectors = 64; | 528 | mmc->max_req_size = 65535; |
530 | 529 | ||
531 | /* | 530 | /* |
532 | * Set the maximum segment size. Since we aren't doing DMA | 531 | * Set the maximum segment size. Since we aren't doing DMA |
533 | * (yet) we are only limited by the data length register. | 532 | * (yet) we are only limited by the data length register. |
534 | */ | 533 | */ |
535 | mmc->max_seg_size = mmc->max_sectors << 9; | 534 | mmc->max_seg_size = mmc->max_req_size; |
535 | |||
536 | /* | ||
537 | * Block size can be up to 2048 bytes, but must be a power of two. | ||
538 | */ | ||
539 | mmc->max_blk_size = 2048; | ||
540 | |||
541 | /* | ||
542 | * No limit on the number of blocks transferred. | ||
543 | */ | ||
544 | mmc->max_blk_count = mmc->max_req_size; | ||
536 | 545 | ||
537 | spin_lock_init(&host->lock); | 546 | spin_lock_init(&host->lock); |
538 | 547 | ||
diff --git a/drivers/mmc/omap.c b/drivers/mmc/omap.c index d30540b27614..1e96a2f65022 100644 --- a/drivers/mmc/omap.c +++ b/drivers/mmc/omap.c | |||
@@ -1099,8 +1099,10 @@ static int __init mmc_omap_probe(struct platform_device *pdev) | |||
1099 | */ | 1099 | */ |
1100 | mmc->max_phys_segs = 32; | 1100 | mmc->max_phys_segs = 32; |
1101 | mmc->max_hw_segs = 32; | 1101 | mmc->max_hw_segs = 32; |
1102 | mmc->max_sectors = 256; /* NBLK max 11-bits, OMAP also limited by DMA */ | 1102 | mmc->max_blk_size = 2048; /* BLEN is 11 bits (+1) */ |
1103 | mmc->max_seg_size = mmc->max_sectors * 512; | 1103 | mmc->max_blk_count = 2048; /* NBLK is 11 bits (+1) */ |
1104 | mmc->max_req_size = mmc->max_blk_size * mmc->max_blk_count; | ||
1105 | mmc->max_seg_size = mmc->max_req_size; | ||
1104 | 1106 | ||
1105 | if (host->power_pin >= 0) { | 1107 | if (host->power_pin >= 0) { |
1106 | if ((ret = omap_request_gpio(host->power_pin)) != 0) { | 1108 | if ((ret = omap_request_gpio(host->power_pin)) != 0) { |
diff --git a/drivers/mmc/pxamci.c b/drivers/mmc/pxamci.c index 6073d998b11f..9774fc68b61a 100644 --- a/drivers/mmc/pxamci.c +++ b/drivers/mmc/pxamci.c | |||
@@ -450,6 +450,16 @@ static int pxamci_probe(struct platform_device *pdev) | |||
450 | */ | 450 | */ |
451 | mmc->max_seg_size = PAGE_SIZE; | 451 | mmc->max_seg_size = PAGE_SIZE; |
452 | 452 | ||
453 | /* | ||
454 | * Block length register is 10 bits. | ||
455 | */ | ||
456 | mmc->max_blk_size = 1023; | ||
457 | |||
458 | /* | ||
459 | * Block count register is 16 bits. | ||
460 | */ | ||
461 | mmc->max_blk_count = 65535; | ||
462 | |||
453 | host = mmc_priv(mmc); | 463 | host = mmc_priv(mmc); |
454 | host->mmc = mmc; | 464 | host->mmc = mmc; |
455 | host->dma = -1; | 465 | host->dma = -1; |
diff --git a/drivers/mmc/sdhci.c b/drivers/mmc/sdhci.c index c2d13d7e9911..7522f76b15ec 100644 --- a/drivers/mmc/sdhci.c +++ b/drivers/mmc/sdhci.c | |||
@@ -22,9 +22,6 @@ | |||
22 | #include "sdhci.h" | 22 | #include "sdhci.h" |
23 | 23 | ||
24 | #define DRIVER_NAME "sdhci" | 24 | #define DRIVER_NAME "sdhci" |
25 | #define DRIVER_VERSION "0.12" | ||
26 | |||
27 | #define BUGMAIL "<sdhci-devel@list.drzeus.cx>" | ||
28 | 25 | ||
29 | #define DBG(f, x...) \ | 26 | #define DBG(f, x...) \ |
30 | pr_debug(DRIVER_NAME " [%s()]: " f, __func__,## x) | 27 | pr_debug(DRIVER_NAME " [%s()]: " f, __func__,## x) |
@@ -37,6 +34,7 @@ static unsigned int debug_quirks = 0; | |||
37 | #define SDHCI_QUIRK_FORCE_DMA (1<<1) | 34 | #define SDHCI_QUIRK_FORCE_DMA (1<<1) |
38 | /* Controller doesn't like some resets when there is no card inserted. */ | 35 | /* Controller doesn't like some resets when there is no card inserted. */ |
39 | #define SDHCI_QUIRK_NO_CARD_NO_RESET (1<<2) | 36 | #define SDHCI_QUIRK_NO_CARD_NO_RESET (1<<2) |
37 | #define SDHCI_QUIRK_SINGLE_POWER_WRITE (1<<3) | ||
40 | 38 | ||
41 | static const struct pci_device_id pci_ids[] __devinitdata = { | 39 | static const struct pci_device_id pci_ids[] __devinitdata = { |
42 | { | 40 | { |
@@ -65,6 +63,14 @@ static const struct pci_device_id pci_ids[] __devinitdata = { | |||
65 | .driver_data = SDHCI_QUIRK_FORCE_DMA, | 63 | .driver_data = SDHCI_QUIRK_FORCE_DMA, |
66 | }, | 64 | }, |
67 | 65 | ||
66 | { | ||
67 | .vendor = PCI_VENDOR_ID_ENE, | ||
68 | .device = PCI_DEVICE_ID_ENE_CB712_SD, | ||
69 | .subvendor = PCI_ANY_ID, | ||
70 | .subdevice = PCI_ANY_ID, | ||
71 | .driver_data = SDHCI_QUIRK_SINGLE_POWER_WRITE, | ||
72 | }, | ||
73 | |||
68 | { /* Generic SD host controller */ | 74 | { /* Generic SD host controller */ |
69 | PCI_DEVICE_CLASS((PCI_CLASS_SYSTEM_SDHCI << 8), 0xFFFF00) | 75 | PCI_DEVICE_CLASS((PCI_CLASS_SYSTEM_SDHCI << 8), 0xFFFF00) |
70 | }, | 76 | }, |
@@ -145,8 +151,7 @@ static void sdhci_reset(struct sdhci_host *host, u8 mask) | |||
145 | /* hw clears the bit when it's done */ | 151 | /* hw clears the bit when it's done */ |
146 | while (readb(host->ioaddr + SDHCI_SOFTWARE_RESET) & mask) { | 152 | while (readb(host->ioaddr + SDHCI_SOFTWARE_RESET) & mask) { |
147 | if (timeout == 0) { | 153 | if (timeout == 0) { |
148 | printk(KERN_ERR "%s: Reset 0x%x never completed. " | 154 | printk(KERN_ERR "%s: Reset 0x%x never completed.\n", |
149 | "Please report this to " BUGMAIL ".\n", | ||
150 | mmc_hostname(host->mmc), (int)mask); | 155 | mmc_hostname(host->mmc), (int)mask); |
151 | sdhci_dumpregs(host); | 156 | sdhci_dumpregs(host); |
152 | return; | 157 | return; |
@@ -197,15 +202,9 @@ static void sdhci_deactivate_led(struct sdhci_host *host) | |||
197 | * * | 202 | * * |
198 | \*****************************************************************************/ | 203 | \*****************************************************************************/ |
199 | 204 | ||
200 | static inline char* sdhci_kmap_sg(struct sdhci_host* host) | 205 | static inline char* sdhci_sg_to_buffer(struct sdhci_host* host) |
201 | { | 206 | { |
202 | host->mapped_sg = kmap_atomic(host->cur_sg->page, KM_BIO_SRC_IRQ); | 207 | return page_address(host->cur_sg->page) + host->cur_sg->offset; |
203 | return host->mapped_sg + host->cur_sg->offset; | ||
204 | } | ||
205 | |||
206 | static inline void sdhci_kunmap_sg(struct sdhci_host* host) | ||
207 | { | ||
208 | kunmap_atomic(host->mapped_sg, KM_BIO_SRC_IRQ); | ||
209 | } | 208 | } |
210 | 209 | ||
211 | static inline int sdhci_next_sg(struct sdhci_host* host) | 210 | static inline int sdhci_next_sg(struct sdhci_host* host) |
@@ -240,7 +239,7 @@ static void sdhci_read_block_pio(struct sdhci_host *host) | |||
240 | chunk_remain = 0; | 239 | chunk_remain = 0; |
241 | data = 0; | 240 | data = 0; |
242 | 241 | ||
243 | buffer = sdhci_kmap_sg(host) + host->offset; | 242 | buffer = sdhci_sg_to_buffer(host) + host->offset; |
244 | 243 | ||
245 | while (blksize) { | 244 | while (blksize) { |
246 | if (chunk_remain == 0) { | 245 | if (chunk_remain == 0) { |
@@ -264,16 +263,13 @@ static void sdhci_read_block_pio(struct sdhci_host *host) | |||
264 | } | 263 | } |
265 | 264 | ||
266 | if (host->remain == 0) { | 265 | if (host->remain == 0) { |
267 | sdhci_kunmap_sg(host); | ||
268 | if (sdhci_next_sg(host) == 0) { | 266 | if (sdhci_next_sg(host) == 0) { |
269 | BUG_ON(blksize != 0); | 267 | BUG_ON(blksize != 0); |
270 | return; | 268 | return; |
271 | } | 269 | } |
272 | buffer = sdhci_kmap_sg(host); | 270 | buffer = sdhci_sg_to_buffer(host); |
273 | } | 271 | } |
274 | } | 272 | } |
275 | |||
276 | sdhci_kunmap_sg(host); | ||
277 | } | 273 | } |
278 | 274 | ||
279 | static void sdhci_write_block_pio(struct sdhci_host *host) | 275 | static void sdhci_write_block_pio(struct sdhci_host *host) |
@@ -290,7 +286,7 @@ static void sdhci_write_block_pio(struct sdhci_host *host) | |||
290 | data = 0; | 286 | data = 0; |
291 | 287 | ||
292 | bytes = 0; | 288 | bytes = 0; |
293 | buffer = sdhci_kmap_sg(host) + host->offset; | 289 | buffer = sdhci_sg_to_buffer(host) + host->offset; |
294 | 290 | ||
295 | while (blksize) { | 291 | while (blksize) { |
296 | size = min(host->size, host->remain); | 292 | size = min(host->size, host->remain); |
@@ -314,16 +310,13 @@ static void sdhci_write_block_pio(struct sdhci_host *host) | |||
314 | } | 310 | } |
315 | 311 | ||
316 | if (host->remain == 0) { | 312 | if (host->remain == 0) { |
317 | sdhci_kunmap_sg(host); | ||
318 | if (sdhci_next_sg(host) == 0) { | 313 | if (sdhci_next_sg(host) == 0) { |
319 | BUG_ON(blksize != 0); | 314 | BUG_ON(blksize != 0); |
320 | return; | 315 | return; |
321 | } | 316 | } |
322 | buffer = sdhci_kmap_sg(host); | 317 | buffer = sdhci_sg_to_buffer(host); |
323 | } | 318 | } |
324 | } | 319 | } |
325 | |||
326 | sdhci_kunmap_sg(host); | ||
327 | } | 320 | } |
328 | 321 | ||
329 | static void sdhci_transfer_pio(struct sdhci_host *host) | 322 | static void sdhci_transfer_pio(struct sdhci_host *host) |
@@ -372,7 +365,7 @@ static void sdhci_prepare_data(struct sdhci_host *host, struct mmc_data *data) | |||
372 | 365 | ||
373 | /* Sanity checks */ | 366 | /* Sanity checks */ |
374 | BUG_ON(data->blksz * data->blocks > 524288); | 367 | BUG_ON(data->blksz * data->blocks > 524288); |
375 | BUG_ON(data->blksz > host->max_block); | 368 | BUG_ON(data->blksz > host->mmc->max_blk_size); |
376 | BUG_ON(data->blocks > 65535); | 369 | BUG_ON(data->blocks > 65535); |
377 | 370 | ||
378 | /* timeout in us */ | 371 | /* timeout in us */ |
@@ -477,12 +470,11 @@ static void sdhci_finish_data(struct sdhci_host *host) | |||
477 | 470 | ||
478 | if ((data->error == MMC_ERR_NONE) && blocks) { | 471 | if ((data->error == MMC_ERR_NONE) && blocks) { |
479 | printk(KERN_ERR "%s: Controller signalled completion even " | 472 | printk(KERN_ERR "%s: Controller signalled completion even " |
480 | "though there were blocks left. Please report this " | 473 | "though there were blocks left.\n", |
481 | "to " BUGMAIL ".\n", mmc_hostname(host->mmc)); | 474 | mmc_hostname(host->mmc)); |
482 | data->error = MMC_ERR_FAILED; | 475 | data->error = MMC_ERR_FAILED; |
483 | } else if (host->size != 0) { | 476 | } else if (host->size != 0) { |
484 | printk(KERN_ERR "%s: %d bytes were left untransferred. " | 477 | printk(KERN_ERR "%s: %d bytes were left untransferred.\n", |
485 | "Please report this to " BUGMAIL ".\n", | ||
486 | mmc_hostname(host->mmc), host->size); | 478 | mmc_hostname(host->mmc), host->size); |
487 | data->error = MMC_ERR_FAILED; | 479 | data->error = MMC_ERR_FAILED; |
488 | } | 480 | } |
@@ -529,8 +521,7 @@ static void sdhci_send_command(struct sdhci_host *host, struct mmc_command *cmd) | |||
529 | while (readl(host->ioaddr + SDHCI_PRESENT_STATE) & mask) { | 521 | while (readl(host->ioaddr + SDHCI_PRESENT_STATE) & mask) { |
530 | if (timeout == 0) { | 522 | if (timeout == 0) { |
531 | printk(KERN_ERR "%s: Controller never released " | 523 | printk(KERN_ERR "%s: Controller never released " |
532 | "inhibit bit(s). Please report this to " | 524 | "inhibit bit(s).\n", mmc_hostname(host->mmc)); |
533 | BUGMAIL ".\n", mmc_hostname(host->mmc)); | ||
534 | sdhci_dumpregs(host); | 525 | sdhci_dumpregs(host); |
535 | cmd->error = MMC_ERR_FAILED; | 526 | cmd->error = MMC_ERR_FAILED; |
536 | tasklet_schedule(&host->finish_tasklet); | 527 | tasklet_schedule(&host->finish_tasklet); |
@@ -551,8 +542,7 @@ static void sdhci_send_command(struct sdhci_host *host, struct mmc_command *cmd) | |||
551 | sdhci_set_transfer_mode(host, cmd->data); | 542 | sdhci_set_transfer_mode(host, cmd->data); |
552 | 543 | ||
553 | if ((cmd->flags & MMC_RSP_136) && (cmd->flags & MMC_RSP_BUSY)) { | 544 | if ((cmd->flags & MMC_RSP_136) && (cmd->flags & MMC_RSP_BUSY)) { |
554 | printk(KERN_ERR "%s: Unsupported response type! " | 545 | printk(KERN_ERR "%s: Unsupported response type!\n", |
555 | "Please report this to " BUGMAIL ".\n", | ||
556 | mmc_hostname(host->mmc)); | 546 | mmc_hostname(host->mmc)); |
557 | cmd->error = MMC_ERR_INVALID; | 547 | cmd->error = MMC_ERR_INVALID; |
558 | tasklet_schedule(&host->finish_tasklet); | 548 | tasklet_schedule(&host->finish_tasklet); |
@@ -650,9 +640,8 @@ static void sdhci_set_clock(struct sdhci_host *host, unsigned int clock) | |||
650 | while (!((clk = readw(host->ioaddr + SDHCI_CLOCK_CONTROL)) | 640 | while (!((clk = readw(host->ioaddr + SDHCI_CLOCK_CONTROL)) |
651 | & SDHCI_CLOCK_INT_STABLE)) { | 641 | & SDHCI_CLOCK_INT_STABLE)) { |
652 | if (timeout == 0) { | 642 | if (timeout == 0) { |
653 | printk(KERN_ERR "%s: Internal clock never stabilised. " | 643 | printk(KERN_ERR "%s: Internal clock never " |
654 | "Please report this to " BUGMAIL ".\n", | 644 | "stabilised.\n", mmc_hostname(host->mmc)); |
655 | mmc_hostname(host->mmc)); | ||
656 | sdhci_dumpregs(host); | 645 | sdhci_dumpregs(host); |
657 | return; | 646 | return; |
658 | } | 647 | } |
@@ -674,10 +663,17 @@ static void sdhci_set_power(struct sdhci_host *host, unsigned short power) | |||
674 | if (host->power == power) | 663 | if (host->power == power) |
675 | return; | 664 | return; |
676 | 665 | ||
677 | writeb(0, host->ioaddr + SDHCI_POWER_CONTROL); | 666 | if (power == (unsigned short)-1) { |
678 | 667 | writeb(0, host->ioaddr + SDHCI_POWER_CONTROL); | |
679 | if (power == (unsigned short)-1) | ||
680 | goto out; | 668 | goto out; |
669 | } | ||
670 | |||
671 | /* | ||
672 | * Spec says that we should clear the power reg before setting | ||
673 | * a new value. Some controllers don't seem to like this though. | ||
674 | */ | ||
675 | if (!(host->chip->quirks & SDHCI_QUIRK_SINGLE_POWER_WRITE)) | ||
676 | writeb(0, host->ioaddr + SDHCI_POWER_CONTROL); | ||
681 | 677 | ||
682 | pwr = SDHCI_POWER_ON; | 678 | pwr = SDHCI_POWER_ON; |
683 | 679 | ||
@@ -895,9 +891,8 @@ static void sdhci_timeout_timer(unsigned long data) | |||
895 | spin_lock_irqsave(&host->lock, flags); | 891 | spin_lock_irqsave(&host->lock, flags); |
896 | 892 | ||
897 | if (host->mrq) { | 893 | if (host->mrq) { |
898 | printk(KERN_ERR "%s: Timeout waiting for hardware interrupt. " | 894 | printk(KERN_ERR "%s: Timeout waiting for hardware " |
899 | "Please report this to " BUGMAIL ".\n", | 895 | "interrupt.\n", mmc_hostname(host->mmc)); |
900 | mmc_hostname(host->mmc)); | ||
901 | sdhci_dumpregs(host); | 896 | sdhci_dumpregs(host); |
902 | 897 | ||
903 | if (host->data) { | 898 | if (host->data) { |
@@ -931,8 +926,6 @@ static void sdhci_cmd_irq(struct sdhci_host *host, u32 intmask) | |||
931 | printk(KERN_ERR "%s: Got command interrupt even though no " | 926 | printk(KERN_ERR "%s: Got command interrupt even though no " |
932 | "command operation was in progress.\n", | 927 | "command operation was in progress.\n", |
933 | mmc_hostname(host->mmc)); | 928 | mmc_hostname(host->mmc)); |
934 | printk(KERN_ERR "%s: Please report this to " BUGMAIL ".\n", | ||
935 | mmc_hostname(host->mmc)); | ||
936 | sdhci_dumpregs(host); | 929 | sdhci_dumpregs(host); |
937 | return; | 930 | return; |
938 | } | 931 | } |
@@ -968,8 +961,6 @@ static void sdhci_data_irq(struct sdhci_host *host, u32 intmask) | |||
968 | printk(KERN_ERR "%s: Got data interrupt even though no " | 961 | printk(KERN_ERR "%s: Got data interrupt even though no " |
969 | "data operation was in progress.\n", | 962 | "data operation was in progress.\n", |
970 | mmc_hostname(host->mmc)); | 963 | mmc_hostname(host->mmc)); |
971 | printk(KERN_ERR "%s: Please report this to " BUGMAIL ".\n", | ||
972 | mmc_hostname(host->mmc)); | ||
973 | sdhci_dumpregs(host); | 964 | sdhci_dumpregs(host); |
974 | 965 | ||
975 | return; | 966 | return; |
@@ -1041,8 +1032,7 @@ static irqreturn_t sdhci_irq(int irq, void *dev_id) | |||
1041 | intmask &= SDHCI_INT_BUS_POWER; | 1032 | intmask &= SDHCI_INT_BUS_POWER; |
1042 | 1033 | ||
1043 | if (intmask) { | 1034 | if (intmask) { |
1044 | printk(KERN_ERR "%s: Unexpected interrupt 0x%08x. Please " | 1035 | printk(KERN_ERR "%s: Unexpected interrupt 0x%08x.\n", |
1045 | "report this to " BUGMAIL ".\n", | ||
1046 | mmc_hostname(host->mmc), intmask); | 1036 | mmc_hostname(host->mmc), intmask); |
1047 | sdhci_dumpregs(host); | 1037 | sdhci_dumpregs(host); |
1048 | 1038 | ||
@@ -1109,7 +1099,9 @@ static int sdhci_resume (struct pci_dev *pdev) | |||
1109 | 1099 | ||
1110 | pci_set_power_state(pdev, PCI_D0); | 1100 | pci_set_power_state(pdev, PCI_D0); |
1111 | pci_restore_state(pdev); | 1101 | pci_restore_state(pdev); |
1112 | pci_enable_device(pdev); | 1102 | ret = pci_enable_device(pdev); |
1103 | if (ret) | ||
1104 | return ret; | ||
1113 | 1105 | ||
1114 | for (i = 0;i < chip->num_slots;i++) { | 1106 | for (i = 0;i < chip->num_slots;i++) { |
1115 | if (!chip->hosts[i]) | 1107 | if (!chip->hosts[i]) |
@@ -1274,15 +1266,6 @@ static int __devinit sdhci_probe_slot(struct pci_dev *pdev, int slot) | |||
1274 | if (caps & SDHCI_TIMEOUT_CLK_UNIT) | 1266 | if (caps & SDHCI_TIMEOUT_CLK_UNIT) |
1275 | host->timeout_clk *= 1000; | 1267 | host->timeout_clk *= 1000; |
1276 | 1268 | ||
1277 | host->max_block = (caps & SDHCI_MAX_BLOCK_MASK) >> SDHCI_MAX_BLOCK_SHIFT; | ||
1278 | if (host->max_block >= 3) { | ||
1279 | printk(KERN_ERR "%s: Invalid maximum block size.\n", | ||
1280 | host->slot_descr); | ||
1281 | ret = -ENODEV; | ||
1282 | goto unmap; | ||
1283 | } | ||
1284 | host->max_block = 512 << host->max_block; | ||
1285 | |||
1286 | /* | 1269 | /* |
1287 | * Set host parameters. | 1270 | * Set host parameters. |
1288 | */ | 1271 | */ |
@@ -1294,9 +1277,9 @@ static int __devinit sdhci_probe_slot(struct pci_dev *pdev, int slot) | |||
1294 | mmc->ocr_avail = 0; | 1277 | mmc->ocr_avail = 0; |
1295 | if (caps & SDHCI_CAN_VDD_330) | 1278 | if (caps & SDHCI_CAN_VDD_330) |
1296 | mmc->ocr_avail |= MMC_VDD_32_33|MMC_VDD_33_34; | 1279 | mmc->ocr_avail |= MMC_VDD_32_33|MMC_VDD_33_34; |
1297 | else if (caps & SDHCI_CAN_VDD_300) | 1280 | if (caps & SDHCI_CAN_VDD_300) |
1298 | mmc->ocr_avail |= MMC_VDD_29_30|MMC_VDD_30_31; | 1281 | mmc->ocr_avail |= MMC_VDD_29_30|MMC_VDD_30_31; |
1299 | else if (caps & SDHCI_CAN_VDD_180) | 1282 | if (caps & SDHCI_CAN_VDD_180) |
1300 | mmc->ocr_avail |= MMC_VDD_17_18|MMC_VDD_18_19; | 1283 | mmc->ocr_avail |= MMC_VDD_17_18|MMC_VDD_18_19; |
1301 | 1284 | ||
1302 | if ((host->max_clk > 25000000) && !(caps & SDHCI_CAN_DO_HISPD)) { | 1285 | if ((host->max_clk > 25000000) && !(caps & SDHCI_CAN_DO_HISPD)) { |
@@ -1326,15 +1309,33 @@ static int __devinit sdhci_probe_slot(struct pci_dev *pdev, int slot) | |||
1326 | 1309 | ||
1327 | /* | 1310 | /* |
1328 | * Maximum number of sectors in one transfer. Limited by DMA boundary | 1311 | * Maximum number of sectors in one transfer. Limited by DMA boundary |
1329 | * size (512KiB), which means (512 KiB/512=) 1024 entries. | 1312 | * size (512KiB). |
1330 | */ | 1313 | */ |
1331 | mmc->max_sectors = 1024; | 1314 | mmc->max_req_size = 524288; |
1332 | 1315 | ||
1333 | /* | 1316 | /* |
1334 | * Maximum segment size. Could be one segment with the maximum number | 1317 | * Maximum segment size. Could be one segment with the maximum number |
1335 | * of sectors. | 1318 | * of bytes. |
1319 | */ | ||
1320 | mmc->max_seg_size = mmc->max_req_size; | ||
1321 | |||
1322 | /* | ||
1323 | * Maximum block size. This varies from controller to controller and | ||
1324 | * is specified in the capabilities register. | ||
1325 | */ | ||
1326 | mmc->max_blk_size = (caps & SDHCI_MAX_BLOCK_MASK) >> SDHCI_MAX_BLOCK_SHIFT; | ||
1327 | if (mmc->max_blk_size >= 3) { | ||
1328 | printk(KERN_ERR "%s: Invalid maximum block size.\n", | ||
1329 | host->slot_descr); | ||
1330 | ret = -ENODEV; | ||
1331 | goto unmap; | ||
1332 | } | ||
1333 | mmc->max_blk_size = 512 << mmc->max_blk_size; | ||
1334 | |||
1335 | /* | ||
1336 | * Maximum block count. | ||
1336 | */ | 1337 | */ |
1337 | mmc->max_seg_size = mmc->max_sectors * 512; | 1338 | mmc->max_blk_count = 65535; |
1338 | 1339 | ||
1339 | /* | 1340 | /* |
1340 | * Init tasklets. | 1341 | * Init tasklets. |
@@ -1513,8 +1514,7 @@ static struct pci_driver sdhci_driver = { | |||
1513 | static int __init sdhci_drv_init(void) | 1514 | static int __init sdhci_drv_init(void) |
1514 | { | 1515 | { |
1515 | printk(KERN_INFO DRIVER_NAME | 1516 | printk(KERN_INFO DRIVER_NAME |
1516 | ": Secure Digital Host Controller Interface driver, " | 1517 | ": Secure Digital Host Controller Interface driver\n"); |
1517 | DRIVER_VERSION "\n"); | ||
1518 | printk(KERN_INFO DRIVER_NAME ": Copyright(c) Pierre Ossman\n"); | 1518 | printk(KERN_INFO DRIVER_NAME ": Copyright(c) Pierre Ossman\n"); |
1519 | 1519 | ||
1520 | return pci_register_driver(&sdhci_driver); | 1520 | return pci_register_driver(&sdhci_driver); |
@@ -1536,7 +1536,6 @@ module_param(debug_quirks, uint, 0444); | |||
1536 | 1536 | ||
1537 | MODULE_AUTHOR("Pierre Ossman <drzeus@drzeus.cx>"); | 1537 | MODULE_AUTHOR("Pierre Ossman <drzeus@drzeus.cx>"); |
1538 | MODULE_DESCRIPTION("Secure Digital Host Controller Interface driver"); | 1538 | MODULE_DESCRIPTION("Secure Digital Host Controller Interface driver"); |
1539 | MODULE_VERSION(DRIVER_VERSION); | ||
1540 | MODULE_LICENSE("GPL"); | 1539 | MODULE_LICENSE("GPL"); |
1541 | 1540 | ||
1542 | MODULE_PARM_DESC(debug_nodma, "Forcefully disable DMA transfers. (default 0)"); | 1541 | MODULE_PARM_DESC(debug_nodma, "Forcefully disable DMA transfers. (default 0)"); |
diff --git a/drivers/mmc/sdhci.h b/drivers/mmc/sdhci.h index f9d1a0a6f03a..e324f0a623dc 100644 --- a/drivers/mmc/sdhci.h +++ b/drivers/mmc/sdhci.h | |||
@@ -174,7 +174,6 @@ struct sdhci_host { | |||
174 | 174 | ||
175 | unsigned int max_clk; /* Max possible freq (MHz) */ | 175 | unsigned int max_clk; /* Max possible freq (MHz) */ |
176 | unsigned int timeout_clk; /* Timeout freq (KHz) */ | 176 | unsigned int timeout_clk; /* Timeout freq (KHz) */ |
177 | unsigned int max_block; /* Max block size (bytes) */ | ||
178 | 177 | ||
179 | unsigned int clock; /* Current clock (MHz) */ | 178 | unsigned int clock; /* Current clock (MHz) */ |
180 | unsigned short power; /* Current voltage */ | 179 | unsigned short power; /* Current voltage */ |
@@ -184,7 +183,6 @@ struct sdhci_host { | |||
184 | struct mmc_data *data; /* Current data request */ | 183 | struct mmc_data *data; /* Current data request */ |
185 | 184 | ||
186 | struct scatterlist *cur_sg; /* We're working on this */ | 185 | struct scatterlist *cur_sg; /* We're working on this */ |
187 | char *mapped_sg; /* This is where it's mapped */ | ||
188 | int num_sg; /* Entries left */ | 186 | int num_sg; /* Entries left */ |
189 | int offset; /* Offset into current sg */ | 187 | int offset; /* Offset into current sg */ |
190 | int remain; /* Bytes left in current */ | 188 | int remain; /* Bytes left in current */ |
diff --git a/drivers/mmc/tifm_sd.c b/drivers/mmc/tifm_sd.c index fa4a52886b97..0581d09c58fc 100644 --- a/drivers/mmc/tifm_sd.c +++ b/drivers/mmc/tifm_sd.c | |||
@@ -17,7 +17,7 @@ | |||
17 | #include <asm/io.h> | 17 | #include <asm/io.h> |
18 | 18 | ||
19 | #define DRIVER_NAME "tifm_sd" | 19 | #define DRIVER_NAME "tifm_sd" |
20 | #define DRIVER_VERSION "0.6" | 20 | #define DRIVER_VERSION "0.7" |
21 | 21 | ||
22 | static int no_dma = 0; | 22 | static int no_dma = 0; |
23 | static int fixed_timeout = 0; | 23 | static int fixed_timeout = 0; |
@@ -36,8 +36,8 @@ module_param(fixed_timeout, bool, 0644); | |||
36 | #define TIFM_MMCSD_INAB 0x0080 /* abort / initialize command */ | 36 | #define TIFM_MMCSD_INAB 0x0080 /* abort / initialize command */ |
37 | #define TIFM_MMCSD_READ 0x8000 | 37 | #define TIFM_MMCSD_READ 0x8000 |
38 | 38 | ||
39 | #define TIFM_MMCSD_DATAMASK 0x001d /* set bits: EOFB, BRS, CB, EOC */ | 39 | #define TIFM_MMCSD_DATAMASK 0x401d /* set bits: CERR, EOFB, BRS, CB, EOC */ |
40 | #define TIFM_MMCSD_ERRMASK 0x41e0 /* set bits: CERR, CCRC, CTO, DCRC, DTO */ | 40 | #define TIFM_MMCSD_ERRMASK 0x01e0 /* set bits: CCRC, CTO, DCRC, DTO */ |
41 | #define TIFM_MMCSD_EOC 0x0001 /* end of command phase */ | 41 | #define TIFM_MMCSD_EOC 0x0001 /* end of command phase */ |
42 | #define TIFM_MMCSD_CB 0x0004 /* card enter busy state */ | 42 | #define TIFM_MMCSD_CB 0x0004 /* card enter busy state */ |
43 | #define TIFM_MMCSD_BRS 0x0008 /* block received/sent */ | 43 | #define TIFM_MMCSD_BRS 0x0008 /* block received/sent */ |
@@ -79,7 +79,6 @@ typedef enum { | |||
79 | 79 | ||
80 | enum { | 80 | enum { |
81 | FIFO_RDY = 0x0001, /* hardware dependent value */ | 81 | FIFO_RDY = 0x0001, /* hardware dependent value */ |
82 | HOST_REG = 0x0002, | ||
83 | EJECT = 0x0004, | 82 | EJECT = 0x0004, |
84 | EJECT_DONE = 0x0008, | 83 | EJECT_DONE = 0x0008, |
85 | CARD_BUSY = 0x0010, | 84 | CARD_BUSY = 0x0010, |
@@ -95,46 +94,53 @@ struct tifm_sd { | |||
95 | card_state_t state; | 94 | card_state_t state; |
96 | unsigned int clk_freq; | 95 | unsigned int clk_freq; |
97 | unsigned int clk_div; | 96 | unsigned int clk_div; |
98 | unsigned long timeout_jiffies; // software timeout - 2 sec | 97 | unsigned long timeout_jiffies; |
99 | 98 | ||
99 | struct tasklet_struct finish_tasklet; | ||
100 | struct timer_list timer; | ||
100 | struct mmc_request *req; | 101 | struct mmc_request *req; |
101 | struct work_struct cmd_handler; | 102 | wait_queue_head_t notify; |
102 | struct delayed_work abort_handler; | ||
103 | wait_queue_head_t can_eject; | ||
104 | 103 | ||
105 | size_t written_blocks; | 104 | size_t written_blocks; |
106 | char *buffer; | ||
107 | size_t buffer_size; | 105 | size_t buffer_size; |
108 | size_t buffer_pos; | 106 | size_t buffer_pos; |
109 | 107 | ||
110 | }; | 108 | }; |
111 | 109 | ||
110 | static char* tifm_sd_data_buffer(struct mmc_data *data) | ||
111 | { | ||
112 | return page_address(data->sg->page) + data->sg->offset; | ||
113 | } | ||
114 | |||
112 | static int tifm_sd_transfer_data(struct tifm_dev *sock, struct tifm_sd *host, | 115 | static int tifm_sd_transfer_data(struct tifm_dev *sock, struct tifm_sd *host, |
113 | unsigned int host_status) | 116 | unsigned int host_status) |
114 | { | 117 | { |
115 | struct mmc_command *cmd = host->req->cmd; | 118 | struct mmc_command *cmd = host->req->cmd; |
116 | unsigned int t_val = 0, cnt = 0; | 119 | unsigned int t_val = 0, cnt = 0; |
120 | char *buffer; | ||
117 | 121 | ||
118 | if (host_status & TIFM_MMCSD_BRS) { | 122 | if (host_status & TIFM_MMCSD_BRS) { |
119 | /* in non-dma rx mode BRS fires when fifo is still not empty */ | 123 | /* in non-dma rx mode BRS fires when fifo is still not empty */ |
120 | if (host->buffer && (cmd->data->flags & MMC_DATA_READ)) { | 124 | if (no_dma && (cmd->data->flags & MMC_DATA_READ)) { |
125 | buffer = tifm_sd_data_buffer(host->req->data); | ||
121 | while (host->buffer_size > host->buffer_pos) { | 126 | while (host->buffer_size > host->buffer_pos) { |
122 | t_val = readl(sock->addr + SOCK_MMCSD_DATA); | 127 | t_val = readl(sock->addr + SOCK_MMCSD_DATA); |
123 | host->buffer[host->buffer_pos++] = t_val & 0xff; | 128 | buffer[host->buffer_pos++] = t_val & 0xff; |
124 | host->buffer[host->buffer_pos++] = | 129 | buffer[host->buffer_pos++] = |
125 | (t_val >> 8) & 0xff; | 130 | (t_val >> 8) & 0xff; |
126 | } | 131 | } |
127 | } | 132 | } |
128 | return 1; | 133 | return 1; |
129 | } else if (host->buffer) { | 134 | } else if (no_dma) { |
135 | buffer = tifm_sd_data_buffer(host->req->data); | ||
130 | if ((cmd->data->flags & MMC_DATA_READ) && | 136 | if ((cmd->data->flags & MMC_DATA_READ) && |
131 | (host_status & TIFM_MMCSD_AF)) { | 137 | (host_status & TIFM_MMCSD_AF)) { |
132 | for (cnt = 0; cnt < TIFM_MMCSD_FIFO_SIZE; cnt++) { | 138 | for (cnt = 0; cnt < TIFM_MMCSD_FIFO_SIZE; cnt++) { |
133 | t_val = readl(sock->addr + SOCK_MMCSD_DATA); | 139 | t_val = readl(sock->addr + SOCK_MMCSD_DATA); |
134 | if (host->buffer_size > host->buffer_pos) { | 140 | if (host->buffer_size > host->buffer_pos) { |
135 | host->buffer[host->buffer_pos++] = | 141 | buffer[host->buffer_pos++] = |
136 | t_val & 0xff; | 142 | t_val & 0xff; |
137 | host->buffer[host->buffer_pos++] = | 143 | buffer[host->buffer_pos++] = |
138 | (t_val >> 8) & 0xff; | 144 | (t_val >> 8) & 0xff; |
139 | } | 145 | } |
140 | } | 146 | } |
@@ -142,11 +148,12 @@ static int tifm_sd_transfer_data(struct tifm_dev *sock, struct tifm_sd *host, | |||
142 | && (host_status & TIFM_MMCSD_AE)) { | 148 | && (host_status & TIFM_MMCSD_AE)) { |
143 | for (cnt = 0; cnt < TIFM_MMCSD_FIFO_SIZE; cnt++) { | 149 | for (cnt = 0; cnt < TIFM_MMCSD_FIFO_SIZE; cnt++) { |
144 | if (host->buffer_size > host->buffer_pos) { | 150 | if (host->buffer_size > host->buffer_pos) { |
145 | t_val = host->buffer[host->buffer_pos++] & 0x00ff; | 151 | t_val = buffer[host->buffer_pos++] |
146 | t_val |= ((host->buffer[host->buffer_pos++]) << 8) | 152 | & 0x00ff; |
147 | & 0xff00; | 153 | t_val |= ((buffer[host->buffer_pos++]) |
154 | << 8) & 0xff00; | ||
148 | writel(t_val, | 155 | writel(t_val, |
149 | sock->addr + SOCK_MMCSD_DATA); | 156 | sock->addr + SOCK_MMCSD_DATA); |
150 | } | 157 | } |
151 | } | 158 | } |
152 | } | 159 | } |
@@ -206,7 +213,7 @@ static void tifm_sd_exec(struct tifm_sd *host, struct mmc_command *cmd) | |||
206 | cmd_mask |= TIFM_MMCSD_READ; | 213 | cmd_mask |= TIFM_MMCSD_READ; |
207 | 214 | ||
208 | dev_dbg(&sock->dev, "executing opcode 0x%x, arg: 0x%x, mask: 0x%x\n", | 215 | dev_dbg(&sock->dev, "executing opcode 0x%x, arg: 0x%x, mask: 0x%x\n", |
209 | cmd->opcode, cmd->arg, cmd_mask); | 216 | cmd->opcode, cmd->arg, cmd_mask); |
210 | 217 | ||
211 | writel((cmd->arg >> 16) & 0xffff, sock->addr + SOCK_MMCSD_ARG_HIGH); | 218 | writel((cmd->arg >> 16) & 0xffff, sock->addr + SOCK_MMCSD_ARG_HIGH); |
212 | writel(cmd->arg & 0xffff, sock->addr + SOCK_MMCSD_ARG_LOW); | 219 | writel(cmd->arg & 0xffff, sock->addr + SOCK_MMCSD_ARG_LOW); |
@@ -235,69 +242,82 @@ change_state: | |||
235 | case IDLE: | 242 | case IDLE: |
236 | return; | 243 | return; |
237 | case CMD: | 244 | case CMD: |
238 | if (host_status & TIFM_MMCSD_EOC) { | 245 | if (host_status & (TIFM_MMCSD_EOC | TIFM_MMCSD_CERR)) { |
239 | tifm_sd_fetch_resp(cmd, sock); | 246 | tifm_sd_fetch_resp(cmd, sock); |
240 | if (cmd->data) { | 247 | if (cmd->data) { |
241 | host->state = BRS; | 248 | host->state = BRS; |
242 | } else | 249 | } else { |
243 | host->state = READY; | 250 | host->state = READY; |
251 | } | ||
244 | goto change_state; | 252 | goto change_state; |
245 | } | 253 | } |
246 | break; | 254 | break; |
247 | case BRS: | 255 | case BRS: |
248 | if (tifm_sd_transfer_data(sock, host, host_status)) { | 256 | if (tifm_sd_transfer_data(sock, host, host_status)) { |
249 | if (!host->req->stop) { | 257 | if (cmd->data->flags & MMC_DATA_WRITE) { |
250 | if (cmd->data->flags & MMC_DATA_WRITE) { | 258 | host->state = CARD; |
251 | host->state = CARD; | 259 | } else { |
260 | if (no_dma) { | ||
261 | if (host->req->stop) { | ||
262 | tifm_sd_exec(host, host->req->stop); | ||
263 | host->state = SCMD; | ||
264 | } else { | ||
265 | host->state = READY; | ||
266 | } | ||
252 | } else { | 267 | } else { |
253 | host->state = | 268 | host->state = FIFO; |
254 | host->buffer ? READY : FIFO; | ||
255 | } | 269 | } |
256 | goto change_state; | ||
257 | } | 270 | } |
258 | tifm_sd_exec(host, host->req->stop); | 271 | goto change_state; |
259 | host->state = SCMD; | ||
260 | } | 272 | } |
261 | break; | 273 | break; |
262 | case SCMD: | 274 | case SCMD: |
263 | if (host_status & TIFM_MMCSD_EOC) { | 275 | if (host_status & TIFM_MMCSD_EOC) { |
264 | tifm_sd_fetch_resp(host->req->stop, sock); | 276 | tifm_sd_fetch_resp(host->req->stop, sock); |
265 | if (cmd->error) { | 277 | host->state = READY; |
266 | host->state = READY; | ||
267 | } else if (cmd->data->flags & MMC_DATA_WRITE) { | ||
268 | host->state = CARD; | ||
269 | } else { | ||
270 | host->state = host->buffer ? READY : FIFO; | ||
271 | } | ||
272 | goto change_state; | 278 | goto change_state; |
273 | } | 279 | } |
274 | break; | 280 | break; |
275 | case CARD: | 281 | case CARD: |
282 | dev_dbg(&sock->dev, "waiting for CARD, have %zd blocks\n", | ||
283 | host->written_blocks); | ||
276 | if (!(host->flags & CARD_BUSY) | 284 | if (!(host->flags & CARD_BUSY) |
277 | && (host->written_blocks == cmd->data->blocks)) { | 285 | && (host->written_blocks == cmd->data->blocks)) { |
278 | host->state = host->buffer ? READY : FIFO; | 286 | if (no_dma) { |
287 | if (host->req->stop) { | ||
288 | tifm_sd_exec(host, host->req->stop); | ||
289 | host->state = SCMD; | ||
290 | } else { | ||
291 | host->state = READY; | ||
292 | } | ||
293 | } else { | ||
294 | host->state = FIFO; | ||
295 | } | ||
279 | goto change_state; | 296 | goto change_state; |
280 | } | 297 | } |
281 | break; | 298 | break; |
282 | case FIFO: | 299 | case FIFO: |
283 | if (host->flags & FIFO_RDY) { | 300 | if (host->flags & FIFO_RDY) { |
284 | host->state = READY; | ||
285 | host->flags &= ~FIFO_RDY; | 301 | host->flags &= ~FIFO_RDY; |
302 | if (host->req->stop) { | ||
303 | tifm_sd_exec(host, host->req->stop); | ||
304 | host->state = SCMD; | ||
305 | } else { | ||
306 | host->state = READY; | ||
307 | } | ||
286 | goto change_state; | 308 | goto change_state; |
287 | } | 309 | } |
288 | break; | 310 | break; |
289 | case READY: | 311 | case READY: |
290 | queue_work(sock->wq, &host->cmd_handler); | 312 | tasklet_schedule(&host->finish_tasklet); |
291 | return; | 313 | return; |
292 | } | 314 | } |
293 | 315 | ||
294 | queue_delayed_work(sock->wq, &host->abort_handler, | ||
295 | host->timeout_jiffies); | ||
296 | } | 316 | } |
297 | 317 | ||
298 | /* Called from interrupt handler */ | 318 | /* Called from interrupt handler */ |
299 | static unsigned int tifm_sd_signal_irq(struct tifm_dev *sock, | 319 | static void tifm_sd_signal_irq(struct tifm_dev *sock, |
300 | unsigned int sock_irq_status) | 320 | unsigned int sock_irq_status) |
301 | { | 321 | { |
302 | struct tifm_sd *host; | 322 | struct tifm_sd *host; |
303 | unsigned int host_status = 0, fifo_status = 0; | 323 | unsigned int host_status = 0, fifo_status = 0; |
@@ -305,7 +325,6 @@ static unsigned int tifm_sd_signal_irq(struct tifm_dev *sock, | |||
305 | 325 | ||
306 | spin_lock(&sock->lock); | 326 | spin_lock(&sock->lock); |
307 | host = mmc_priv((struct mmc_host*)tifm_get_drvdata(sock)); | 327 | host = mmc_priv((struct mmc_host*)tifm_get_drvdata(sock)); |
308 | cancel_delayed_work(&host->abort_handler); | ||
309 | 328 | ||
310 | if (sock_irq_status & FIFO_EVENT) { | 329 | if (sock_irq_status & FIFO_EVENT) { |
311 | fifo_status = readl(sock->addr + SOCK_DMA_FIFO_STATUS); | 330 | fifo_status = readl(sock->addr + SOCK_DMA_FIFO_STATUS); |
@@ -318,19 +337,14 @@ static unsigned int tifm_sd_signal_irq(struct tifm_dev *sock, | |||
318 | host_status = readl(sock->addr + SOCK_MMCSD_STATUS); | 337 | host_status = readl(sock->addr + SOCK_MMCSD_STATUS); |
319 | writel(host_status, sock->addr + SOCK_MMCSD_STATUS); | 338 | writel(host_status, sock->addr + SOCK_MMCSD_STATUS); |
320 | 339 | ||
321 | if (!(host->flags & HOST_REG)) | ||
322 | queue_work(sock->wq, &host->cmd_handler); | ||
323 | if (!host->req) | 340 | if (!host->req) |
324 | goto done; | 341 | goto done; |
325 | 342 | ||
326 | if (host_status & TIFM_MMCSD_ERRMASK) { | 343 | if (host_status & TIFM_MMCSD_ERRMASK) { |
327 | if (host_status & TIFM_MMCSD_CERR) | 344 | if (host_status & (TIFM_MMCSD_CTO | TIFM_MMCSD_DTO)) |
328 | error_code = MMC_ERR_FAILED; | ||
329 | else if (host_status & | ||
330 | (TIFM_MMCSD_CTO | TIFM_MMCSD_DTO)) | ||
331 | error_code = MMC_ERR_TIMEOUT; | 345 | error_code = MMC_ERR_TIMEOUT; |
332 | else if (host_status & | 346 | else if (host_status |
333 | (TIFM_MMCSD_CCRC | TIFM_MMCSD_DCRC)) | 347 | & (TIFM_MMCSD_CCRC | TIFM_MMCSD_DCRC)) |
334 | error_code = MMC_ERR_BADCRC; | 348 | error_code = MMC_ERR_BADCRC; |
335 | 349 | ||
336 | writel(TIFM_FIFO_INT_SETALL, | 350 | writel(TIFM_FIFO_INT_SETALL, |
@@ -340,12 +354,11 @@ static unsigned int tifm_sd_signal_irq(struct tifm_dev *sock, | |||
340 | if (host->req->stop) { | 354 | if (host->req->stop) { |
341 | if (host->state == SCMD) { | 355 | if (host->state == SCMD) { |
342 | host->req->stop->error = error_code; | 356 | host->req->stop->error = error_code; |
343 | } else if(host->state == BRS) { | 357 | } else if (host->state == BRS |
358 | || host->state == CARD | ||
359 | || host->state == FIFO) { | ||
344 | host->req->cmd->error = error_code; | 360 | host->req->cmd->error = error_code; |
345 | tifm_sd_exec(host, host->req->stop); | 361 | tifm_sd_exec(host, host->req->stop); |
346 | queue_delayed_work(sock->wq, | ||
347 | &host->abort_handler, | ||
348 | host->timeout_jiffies); | ||
349 | host->state = SCMD; | 362 | host->state = SCMD; |
350 | goto done; | 363 | goto done; |
351 | } else { | 364 | } else { |
@@ -359,8 +372,8 @@ static unsigned int tifm_sd_signal_irq(struct tifm_dev *sock, | |||
359 | 372 | ||
360 | if (host_status & TIFM_MMCSD_CB) | 373 | if (host_status & TIFM_MMCSD_CB) |
361 | host->flags |= CARD_BUSY; | 374 | host->flags |= CARD_BUSY; |
362 | if ((host_status & TIFM_MMCSD_EOFB) && | 375 | if ((host_status & TIFM_MMCSD_EOFB) |
363 | (host->flags & CARD_BUSY)) { | 376 | && (host->flags & CARD_BUSY)) { |
364 | host->written_blocks++; | 377 | host->written_blocks++; |
365 | host->flags &= ~CARD_BUSY; | 378 | host->flags &= ~CARD_BUSY; |
366 | } | 379 | } |
@@ -370,22 +383,22 @@ static unsigned int tifm_sd_signal_irq(struct tifm_dev *sock, | |||
370 | tifm_sd_process_cmd(sock, host, host_status); | 383 | tifm_sd_process_cmd(sock, host, host_status); |
371 | done: | 384 | done: |
372 | dev_dbg(&sock->dev, "host_status %x, fifo_status %x\n", | 385 | dev_dbg(&sock->dev, "host_status %x, fifo_status %x\n", |
373 | host_status, fifo_status); | 386 | host_status, fifo_status); |
374 | spin_unlock(&sock->lock); | 387 | spin_unlock(&sock->lock); |
375 | return sock_irq_status; | ||
376 | } | 388 | } |
377 | 389 | ||
378 | static void tifm_sd_prepare_data(struct tifm_sd *card, struct mmc_command *cmd) | 390 | static void tifm_sd_prepare_data(struct tifm_sd *host, struct mmc_command *cmd) |
379 | { | 391 | { |
380 | struct tifm_dev *sock = card->dev; | 392 | struct tifm_dev *sock = host->dev; |
381 | unsigned int dest_cnt; | 393 | unsigned int dest_cnt; |
382 | 394 | ||
383 | /* DMA style IO */ | 395 | /* DMA style IO */ |
384 | 396 | dev_dbg(&sock->dev, "setting dma for %d blocks\n", | |
397 | cmd->data->blocks); | ||
385 | writel(TIFM_FIFO_INT_SETALL, | 398 | writel(TIFM_FIFO_INT_SETALL, |
386 | sock->addr + SOCK_DMA_FIFO_INT_ENABLE_CLEAR); | 399 | sock->addr + SOCK_DMA_FIFO_INT_ENABLE_CLEAR); |
387 | writel(ilog2(cmd->data->blksz) - 2, | 400 | writel(ilog2(cmd->data->blksz) - 2, |
388 | sock->addr + SOCK_FIFO_PAGE_SIZE); | 401 | sock->addr + SOCK_FIFO_PAGE_SIZE); |
389 | writel(TIFM_FIFO_ENABLE, sock->addr + SOCK_FIFO_CONTROL); | 402 | writel(TIFM_FIFO_ENABLE, sock->addr + SOCK_FIFO_CONTROL); |
390 | writel(TIFM_FIFO_INTMASK, sock->addr + SOCK_DMA_FIFO_INT_ENABLE_SET); | 403 | writel(TIFM_FIFO_INTMASK, sock->addr + SOCK_DMA_FIFO_INT_ENABLE_SET); |
391 | 404 | ||
@@ -399,7 +412,7 @@ static void tifm_sd_prepare_data(struct tifm_sd *card, struct mmc_command *cmd) | |||
399 | if (cmd->data->flags & MMC_DATA_WRITE) { | 412 | if (cmd->data->flags & MMC_DATA_WRITE) { |
400 | writel(TIFM_MMCSD_TXDE, sock->addr + SOCK_MMCSD_BUFFER_CONFIG); | 413 | writel(TIFM_MMCSD_TXDE, sock->addr + SOCK_MMCSD_BUFFER_CONFIG); |
401 | writel(dest_cnt | TIFM_DMA_TX | TIFM_DMA_EN, | 414 | writel(dest_cnt | TIFM_DMA_TX | TIFM_DMA_EN, |
402 | sock->addr + SOCK_DMA_CONTROL); | 415 | sock->addr + SOCK_DMA_CONTROL); |
403 | } else { | 416 | } else { |
404 | writel(TIFM_MMCSD_RXDE, sock->addr + SOCK_MMCSD_BUFFER_CONFIG); | 417 | writel(TIFM_MMCSD_RXDE, sock->addr + SOCK_MMCSD_BUFFER_CONFIG); |
405 | writel(dest_cnt | TIFM_DMA_EN, sock->addr + SOCK_DMA_CONTROL); | 418 | writel(dest_cnt | TIFM_DMA_EN, sock->addr + SOCK_DMA_CONTROL); |
@@ -407,7 +420,7 @@ static void tifm_sd_prepare_data(struct tifm_sd *card, struct mmc_command *cmd) | |||
407 | } | 420 | } |
408 | 421 | ||
409 | static void tifm_sd_set_data_timeout(struct tifm_sd *host, | 422 | static void tifm_sd_set_data_timeout(struct tifm_sd *host, |
410 | struct mmc_data *data) | 423 | struct mmc_data *data) |
411 | { | 424 | { |
412 | struct tifm_dev *sock = host->dev; | 425 | struct tifm_dev *sock = host->dev; |
413 | unsigned int data_timeout = data->timeout_clks; | 426 | unsigned int data_timeout = data->timeout_clks; |
@@ -416,22 +429,21 @@ static void tifm_sd_set_data_timeout(struct tifm_sd *host, | |||
416 | return; | 429 | return; |
417 | 430 | ||
418 | data_timeout += data->timeout_ns / | 431 | data_timeout += data->timeout_ns / |
419 | ((1000000000 / host->clk_freq) * host->clk_div); | 432 | ((1000000000UL / host->clk_freq) * host->clk_div); |
420 | data_timeout *= 10; // call it fudge factor for now | ||
421 | 433 | ||
422 | if (data_timeout < 0xffff) { | 434 | if (data_timeout < 0xffff) { |
423 | writel((~TIFM_MMCSD_DPE) & | ||
424 | readl(sock->addr + SOCK_MMCSD_SDIO_MODE_CONFIG), | ||
425 | sock->addr + SOCK_MMCSD_SDIO_MODE_CONFIG); | ||
426 | writel(data_timeout, sock->addr + SOCK_MMCSD_DATA_TO); | 435 | writel(data_timeout, sock->addr + SOCK_MMCSD_DATA_TO); |
436 | writel((~TIFM_MMCSD_DPE) | ||
437 | & readl(sock->addr + SOCK_MMCSD_SDIO_MODE_CONFIG), | ||
438 | sock->addr + SOCK_MMCSD_SDIO_MODE_CONFIG); | ||
427 | } else { | 439 | } else { |
428 | writel(TIFM_MMCSD_DPE | | ||
429 | readl(sock->addr + SOCK_MMCSD_SDIO_MODE_CONFIG), | ||
430 | sock->addr + SOCK_MMCSD_SDIO_MODE_CONFIG); | ||
431 | data_timeout = (data_timeout >> 10) + 1; | 440 | data_timeout = (data_timeout >> 10) + 1; |
432 | if(data_timeout > 0xffff) | 441 | if (data_timeout > 0xffff) |
433 | data_timeout = 0; /* set to unlimited */ | 442 | data_timeout = 0; /* set to unlimited */ |
434 | writel(data_timeout, sock->addr + SOCK_MMCSD_DATA_TO); | 443 | writel(data_timeout, sock->addr + SOCK_MMCSD_DATA_TO); |
444 | writel(TIFM_MMCSD_DPE | ||
445 | | readl(sock->addr + SOCK_MMCSD_SDIO_MODE_CONFIG), | ||
446 | sock->addr + SOCK_MMCSD_SDIO_MODE_CONFIG); | ||
435 | } | 447 | } |
436 | } | 448 | } |
437 | 449 | ||
@@ -474,11 +486,10 @@ static void tifm_sd_request(struct mmc_host *mmc, struct mmc_request *mrq) | |||
474 | } | 486 | } |
475 | 487 | ||
476 | host->req = mrq; | 488 | host->req = mrq; |
489 | mod_timer(&host->timer, jiffies + host->timeout_jiffies); | ||
477 | host->state = CMD; | 490 | host->state = CMD; |
478 | queue_delayed_work(sock->wq, &host->abort_handler, | ||
479 | host->timeout_jiffies); | ||
480 | writel(TIFM_CTRL_LED | readl(sock->addr + SOCK_CONTROL), | 491 | writel(TIFM_CTRL_LED | readl(sock->addr + SOCK_CONTROL), |
481 | sock->addr + SOCK_CONTROL); | 492 | sock->addr + SOCK_CONTROL); |
482 | tifm_sd_exec(host, mrq->cmd); | 493 | tifm_sd_exec(host, mrq->cmd); |
483 | spin_unlock_irqrestore(&sock->lock, flags); | 494 | spin_unlock_irqrestore(&sock->lock, flags); |
484 | return; | 495 | return; |
@@ -493,9 +504,9 @@ err_out: | |||
493 | mmc_request_done(mmc, mrq); | 504 | mmc_request_done(mmc, mrq); |
494 | } | 505 | } |
495 | 506 | ||
496 | static void tifm_sd_end_cmd(struct work_struct *work) | 507 | static void tifm_sd_end_cmd(unsigned long data) |
497 | { | 508 | { |
498 | struct tifm_sd *host = container_of(work, struct tifm_sd, cmd_handler); | 509 | struct tifm_sd *host = (struct tifm_sd*)data; |
499 | struct tifm_dev *sock = host->dev; | 510 | struct tifm_dev *sock = host->dev; |
500 | struct mmc_host *mmc = tifm_get_drvdata(sock); | 511 | struct mmc_host *mmc = tifm_get_drvdata(sock); |
501 | struct mmc_request *mrq; | 512 | struct mmc_request *mrq; |
@@ -504,6 +515,7 @@ static void tifm_sd_end_cmd(struct work_struct *work) | |||
504 | 515 | ||
505 | spin_lock_irqsave(&sock->lock, flags); | 516 | spin_lock_irqsave(&sock->lock, flags); |
506 | 517 | ||
518 | del_timer(&host->timer); | ||
507 | mrq = host->req; | 519 | mrq = host->req; |
508 | host->req = NULL; | 520 | host->req = NULL; |
509 | host->state = IDLE; | 521 | host->state = IDLE; |
@@ -517,8 +529,8 @@ static void tifm_sd_end_cmd(struct work_struct *work) | |||
517 | r_data = mrq->cmd->data; | 529 | r_data = mrq->cmd->data; |
518 | if (r_data) { | 530 | if (r_data) { |
519 | if (r_data->flags & MMC_DATA_WRITE) { | 531 | if (r_data->flags & MMC_DATA_WRITE) { |
520 | r_data->bytes_xfered = host->written_blocks * | 532 | r_data->bytes_xfered = host->written_blocks |
521 | r_data->blksz; | 533 | * r_data->blksz; |
522 | } else { | 534 | } else { |
523 | r_data->bytes_xfered = r_data->blocks - | 535 | r_data->bytes_xfered = r_data->blocks - |
524 | readl(sock->addr + SOCK_MMCSD_NUM_BLOCKS) - 1; | 536 | readl(sock->addr + SOCK_MMCSD_NUM_BLOCKS) - 1; |
@@ -532,7 +544,7 @@ static void tifm_sd_end_cmd(struct work_struct *work) | |||
532 | } | 544 | } |
533 | 545 | ||
534 | writel((~TIFM_CTRL_LED) & readl(sock->addr + SOCK_CONTROL), | 546 | writel((~TIFM_CTRL_LED) & readl(sock->addr + SOCK_CONTROL), |
535 | sock->addr + SOCK_CONTROL); | 547 | sock->addr + SOCK_CONTROL); |
536 | 548 | ||
537 | spin_unlock_irqrestore(&sock->lock, flags); | 549 | spin_unlock_irqrestore(&sock->lock, flags); |
538 | mmc_request_done(mmc, mrq); | 550 | mmc_request_done(mmc, mrq); |
@@ -544,15 +556,6 @@ static void tifm_sd_request_nodma(struct mmc_host *mmc, struct mmc_request *mrq) | |||
544 | struct tifm_dev *sock = host->dev; | 556 | struct tifm_dev *sock = host->dev; |
545 | unsigned long flags; | 557 | unsigned long flags; |
546 | struct mmc_data *r_data = mrq->cmd->data; | 558 | struct mmc_data *r_data = mrq->cmd->data; |
547 | char *t_buffer = NULL; | ||
548 | |||
549 | if (r_data) { | ||
550 | t_buffer = kmap(r_data->sg->page); | ||
551 | if (!t_buffer) { | ||
552 | printk(KERN_ERR DRIVER_NAME ": kmap failed\n"); | ||
553 | goto err_out; | ||
554 | } | ||
555 | } | ||
556 | 559 | ||
557 | spin_lock_irqsave(&sock->lock, flags); | 560 | spin_lock_irqsave(&sock->lock, flags); |
558 | if (host->flags & EJECT) { | 561 | if (host->flags & EJECT) { |
@@ -569,15 +572,14 @@ static void tifm_sd_request_nodma(struct mmc_host *mmc, struct mmc_request *mrq) | |||
569 | if (r_data) { | 572 | if (r_data) { |
570 | tifm_sd_set_data_timeout(host, r_data); | 573 | tifm_sd_set_data_timeout(host, r_data); |
571 | 574 | ||
572 | host->buffer = t_buffer + r_data->sg->offset; | 575 | host->buffer_size = mrq->cmd->data->blocks |
573 | host->buffer_size = mrq->cmd->data->blocks * | 576 | * mrq->cmd->data->blksz; |
574 | mrq->cmd->data->blksz; | ||
575 | 577 | ||
576 | writel(TIFM_MMCSD_BUFINT | | 578 | writel(TIFM_MMCSD_BUFINT |
577 | readl(sock->addr + SOCK_MMCSD_INT_ENABLE), | 579 | | readl(sock->addr + SOCK_MMCSD_INT_ENABLE), |
578 | sock->addr + SOCK_MMCSD_INT_ENABLE); | 580 | sock->addr + SOCK_MMCSD_INT_ENABLE); |
579 | writel(((TIFM_MMCSD_FIFO_SIZE - 1) << 8) | | 581 | writel(((TIFM_MMCSD_FIFO_SIZE - 1) << 8) |
580 | (TIFM_MMCSD_FIFO_SIZE - 1), | 582 | | (TIFM_MMCSD_FIFO_SIZE - 1), |
581 | sock->addr + SOCK_MMCSD_BUFFER_CONFIG); | 583 | sock->addr + SOCK_MMCSD_BUFFER_CONFIG); |
582 | 584 | ||
583 | host->written_blocks = 0; | 585 | host->written_blocks = 0; |
@@ -588,26 +590,22 @@ static void tifm_sd_request_nodma(struct mmc_host *mmc, struct mmc_request *mrq) | |||
588 | } | 590 | } |
589 | 591 | ||
590 | host->req = mrq; | 592 | host->req = mrq; |
593 | mod_timer(&host->timer, jiffies + host->timeout_jiffies); | ||
591 | host->state = CMD; | 594 | host->state = CMD; |
592 | queue_delayed_work(sock->wq, &host->abort_handler, | ||
593 | host->timeout_jiffies); | ||
594 | writel(TIFM_CTRL_LED | readl(sock->addr + SOCK_CONTROL), | 595 | writel(TIFM_CTRL_LED | readl(sock->addr + SOCK_CONTROL), |
595 | sock->addr + SOCK_CONTROL); | 596 | sock->addr + SOCK_CONTROL); |
596 | tifm_sd_exec(host, mrq->cmd); | 597 | tifm_sd_exec(host, mrq->cmd); |
597 | spin_unlock_irqrestore(&sock->lock, flags); | 598 | spin_unlock_irqrestore(&sock->lock, flags); |
598 | return; | 599 | return; |
599 | 600 | ||
600 | err_out: | 601 | err_out: |
601 | if (t_buffer) | ||
602 | kunmap(r_data->sg->page); | ||
603 | |||
604 | mrq->cmd->error = MMC_ERR_TIMEOUT; | 602 | mrq->cmd->error = MMC_ERR_TIMEOUT; |
605 | mmc_request_done(mmc, mrq); | 603 | mmc_request_done(mmc, mrq); |
606 | } | 604 | } |
607 | 605 | ||
608 | static void tifm_sd_end_cmd_nodma(struct work_struct *work) | 606 | static void tifm_sd_end_cmd_nodma(unsigned long data) |
609 | { | 607 | { |
610 | struct tifm_sd *host = container_of(work, struct tifm_sd, cmd_handler); | 608 | struct tifm_sd *host = (struct tifm_sd*)data; |
611 | struct tifm_dev *sock = host->dev; | 609 | struct tifm_dev *sock = host->dev; |
612 | struct mmc_host *mmc = tifm_get_drvdata(sock); | 610 | struct mmc_host *mmc = tifm_get_drvdata(sock); |
613 | struct mmc_request *mrq; | 611 | struct mmc_request *mrq; |
@@ -616,6 +614,7 @@ static void tifm_sd_end_cmd_nodma(struct work_struct *work) | |||
616 | 614 | ||
617 | spin_lock_irqsave(&sock->lock, flags); | 615 | spin_lock_irqsave(&sock->lock, flags); |
618 | 616 | ||
617 | del_timer(&host->timer); | ||
619 | mrq = host->req; | 618 | mrq = host->req; |
620 | host->req = NULL; | 619 | host->req = NULL; |
621 | host->state = IDLE; | 620 | host->state = IDLE; |
@@ -633,8 +632,8 @@ static void tifm_sd_end_cmd_nodma(struct work_struct *work) | |||
633 | sock->addr + SOCK_MMCSD_INT_ENABLE); | 632 | sock->addr + SOCK_MMCSD_INT_ENABLE); |
634 | 633 | ||
635 | if (r_data->flags & MMC_DATA_WRITE) { | 634 | if (r_data->flags & MMC_DATA_WRITE) { |
636 | r_data->bytes_xfered = host->written_blocks * | 635 | r_data->bytes_xfered = host->written_blocks |
637 | r_data->blksz; | 636 | * r_data->blksz; |
638 | } else { | 637 | } else { |
639 | r_data->bytes_xfered = r_data->blocks - | 638 | r_data->bytes_xfered = r_data->blocks - |
640 | readl(sock->addr + SOCK_MMCSD_NUM_BLOCKS) - 1; | 639 | readl(sock->addr + SOCK_MMCSD_NUM_BLOCKS) - 1; |
@@ -642,29 +641,44 @@ static void tifm_sd_end_cmd_nodma(struct work_struct *work) | |||
642 | r_data->bytes_xfered += r_data->blksz - | 641 | r_data->bytes_xfered += r_data->blksz - |
643 | readl(sock->addr + SOCK_MMCSD_BLOCK_LEN) + 1; | 642 | readl(sock->addr + SOCK_MMCSD_BLOCK_LEN) + 1; |
644 | } | 643 | } |
645 | host->buffer = NULL; | ||
646 | host->buffer_pos = 0; | 644 | host->buffer_pos = 0; |
647 | host->buffer_size = 0; | 645 | host->buffer_size = 0; |
648 | } | 646 | } |
649 | 647 | ||
650 | writel((~TIFM_CTRL_LED) & readl(sock->addr + SOCK_CONTROL), | 648 | writel((~TIFM_CTRL_LED) & readl(sock->addr + SOCK_CONTROL), |
651 | sock->addr + SOCK_CONTROL); | 649 | sock->addr + SOCK_CONTROL); |
652 | 650 | ||
653 | spin_unlock_irqrestore(&sock->lock, flags); | 651 | spin_unlock_irqrestore(&sock->lock, flags); |
654 | 652 | ||
655 | if (r_data) | ||
656 | kunmap(r_data->sg->page); | ||
657 | |||
658 | mmc_request_done(mmc, mrq); | 653 | mmc_request_done(mmc, mrq); |
659 | } | 654 | } |
660 | 655 | ||
661 | static void tifm_sd_abort(struct work_struct *work) | 656 | static void tifm_sd_terminate(struct tifm_sd *host) |
657 | { | ||
658 | struct tifm_dev *sock = host->dev; | ||
659 | unsigned long flags; | ||
660 | |||
661 | writel(0, sock->addr + SOCK_MMCSD_INT_ENABLE); | ||
662 | mmiowb(); | ||
663 | spin_lock_irqsave(&sock->lock, flags); | ||
664 | host->flags |= EJECT; | ||
665 | if (host->req) { | ||
666 | writel(TIFM_FIFO_INT_SETALL, | ||
667 | sock->addr + SOCK_DMA_FIFO_INT_ENABLE_CLEAR); | ||
668 | writel(0, sock->addr + SOCK_DMA_FIFO_INT_ENABLE_SET); | ||
669 | tasklet_schedule(&host->finish_tasklet); | ||
670 | } | ||
671 | spin_unlock_irqrestore(&sock->lock, flags); | ||
672 | } | ||
673 | |||
674 | static void tifm_sd_abort(unsigned long data) | ||
662 | { | 675 | { |
663 | struct tifm_sd *host = | 676 | struct tifm_sd *host = (struct tifm_sd*)data; |
664 | container_of(work, struct tifm_sd, abort_handler.work); | ||
665 | 677 | ||
666 | printk(KERN_ERR DRIVER_NAME | 678 | printk(KERN_ERR DRIVER_NAME |
667 | ": card failed to respond for a long period of time"); | 679 | ": card failed to respond for a long period of time"); |
680 | |||
681 | tifm_sd_terminate(host); | ||
668 | tifm_eject(host->dev); | 682 | tifm_eject(host->dev); |
669 | } | 683 | } |
670 | 684 | ||
@@ -683,9 +697,9 @@ static void tifm_sd_ios(struct mmc_host *mmc, struct mmc_ios *ios) | |||
683 | writel(TIFM_MMCSD_4BBUS | readl(sock->addr + SOCK_MMCSD_CONFIG), | 697 | writel(TIFM_MMCSD_4BBUS | readl(sock->addr + SOCK_MMCSD_CONFIG), |
684 | sock->addr + SOCK_MMCSD_CONFIG); | 698 | sock->addr + SOCK_MMCSD_CONFIG); |
685 | } else { | 699 | } else { |
686 | writel((~TIFM_MMCSD_4BBUS) & | 700 | writel((~TIFM_MMCSD_4BBUS) |
687 | readl(sock->addr + SOCK_MMCSD_CONFIG), | 701 | & readl(sock->addr + SOCK_MMCSD_CONFIG), |
688 | sock->addr + SOCK_MMCSD_CONFIG); | 702 | sock->addr + SOCK_MMCSD_CONFIG); |
689 | } | 703 | } |
690 | 704 | ||
691 | if (ios->clock) { | 705 | if (ios->clock) { |
@@ -704,23 +718,24 @@ static void tifm_sd_ios(struct mmc_host *mmc, struct mmc_ios *ios) | |||
704 | if ((20000000 / clk_div1) > (24000000 / clk_div2)) { | 718 | if ((20000000 / clk_div1) > (24000000 / clk_div2)) { |
705 | host->clk_freq = 20000000; | 719 | host->clk_freq = 20000000; |
706 | host->clk_div = clk_div1; | 720 | host->clk_div = clk_div1; |
707 | writel((~TIFM_CTRL_FAST_CLK) & | 721 | writel((~TIFM_CTRL_FAST_CLK) |
708 | readl(sock->addr + SOCK_CONTROL), | 722 | & readl(sock->addr + SOCK_CONTROL), |
709 | sock->addr + SOCK_CONTROL); | 723 | sock->addr + SOCK_CONTROL); |
710 | } else { | 724 | } else { |
711 | host->clk_freq = 24000000; | 725 | host->clk_freq = 24000000; |
712 | host->clk_div = clk_div2; | 726 | host->clk_div = clk_div2; |
713 | writel(TIFM_CTRL_FAST_CLK | | 727 | writel(TIFM_CTRL_FAST_CLK |
714 | readl(sock->addr + SOCK_CONTROL), | 728 | | readl(sock->addr + SOCK_CONTROL), |
715 | sock->addr + SOCK_CONTROL); | 729 | sock->addr + SOCK_CONTROL); |
716 | } | 730 | } |
717 | } else { | 731 | } else { |
718 | host->clk_div = 0; | 732 | host->clk_div = 0; |
719 | } | 733 | } |
720 | host->clk_div &= TIFM_MMCSD_CLKMASK; | 734 | host->clk_div &= TIFM_MMCSD_CLKMASK; |
721 | writel(host->clk_div | ((~TIFM_MMCSD_CLKMASK) & | 735 | writel(host->clk_div |
722 | readl(sock->addr + SOCK_MMCSD_CONFIG)), | 736 | | ((~TIFM_MMCSD_CLKMASK) |
723 | sock->addr + SOCK_MMCSD_CONFIG); | 737 | & readl(sock->addr + SOCK_MMCSD_CONFIG)), |
738 | sock->addr + SOCK_MMCSD_CONFIG); | ||
724 | 739 | ||
725 | if (ios->bus_mode == MMC_BUSMODE_OPENDRAIN) | 740 | if (ios->bus_mode == MMC_BUSMODE_OPENDRAIN) |
726 | host->flags |= OPENDRAIN; | 741 | host->flags |= OPENDRAIN; |
@@ -734,7 +749,7 @@ static void tifm_sd_ios(struct mmc_host *mmc, struct mmc_ios *ios) | |||
734 | // allow removal. | 749 | // allow removal. |
735 | if ((host->flags & EJECT) && ios->power_mode == MMC_POWER_OFF) { | 750 | if ((host->flags & EJECT) && ios->power_mode == MMC_POWER_OFF) { |
736 | host->flags |= EJECT_DONE; | 751 | host->flags |= EJECT_DONE; |
737 | wake_up_all(&host->can_eject); | 752 | wake_up_all(&host->notify); |
738 | } | 753 | } |
739 | 754 | ||
740 | spin_unlock_irqrestore(&sock->lock, flags); | 755 | spin_unlock_irqrestore(&sock->lock, flags); |
@@ -762,20 +777,67 @@ static struct mmc_host_ops tifm_sd_ops = { | |||
762 | .get_ro = tifm_sd_ro | 777 | .get_ro = tifm_sd_ro |
763 | }; | 778 | }; |
764 | 779 | ||
765 | static void tifm_sd_register_host(struct work_struct *work) | 780 | static int tifm_sd_initialize_host(struct tifm_sd *host) |
766 | { | 781 | { |
767 | struct tifm_sd *host = container_of(work, struct tifm_sd, cmd_handler); | 782 | int rc; |
783 | unsigned int host_status = 0; | ||
768 | struct tifm_dev *sock = host->dev; | 784 | struct tifm_dev *sock = host->dev; |
769 | struct mmc_host *mmc = tifm_get_drvdata(sock); | ||
770 | unsigned long flags; | ||
771 | 785 | ||
772 | spin_lock_irqsave(&sock->lock, flags); | 786 | writel(0, sock->addr + SOCK_MMCSD_INT_ENABLE); |
773 | host->flags |= HOST_REG; | 787 | mmiowb(); |
774 | PREPARE_WORK(&host->cmd_handler, | 788 | host->clk_div = 61; |
775 | no_dma ? tifm_sd_end_cmd_nodma : tifm_sd_end_cmd); | 789 | host->clk_freq = 20000000; |
776 | spin_unlock_irqrestore(&sock->lock, flags); | 790 | writel(TIFM_MMCSD_RESET, sock->addr + SOCK_MMCSD_SYSTEM_CONTROL); |
777 | dev_dbg(&sock->dev, "adding host\n"); | 791 | writel(host->clk_div | TIFM_MMCSD_POWER, |
778 | mmc_add_host(mmc); | 792 | sock->addr + SOCK_MMCSD_CONFIG); |
793 | |||
794 | /* wait up to 0.51 sec for reset */ | ||
795 | for (rc = 2; rc <= 256; rc <<= 1) { | ||
796 | if (1 & readl(sock->addr + SOCK_MMCSD_SYSTEM_STATUS)) { | ||
797 | rc = 0; | ||
798 | break; | ||
799 | } | ||
800 | msleep(rc); | ||
801 | } | ||
802 | |||
803 | if (rc) { | ||
804 | printk(KERN_ERR DRIVER_NAME | ||
805 | ": controller failed to reset\n"); | ||
806 | return -ENODEV; | ||
807 | } | ||
808 | |||
809 | writel(0, sock->addr + SOCK_MMCSD_NUM_BLOCKS); | ||
810 | writel(host->clk_div | TIFM_MMCSD_POWER, | ||
811 | sock->addr + SOCK_MMCSD_CONFIG); | ||
812 | writel(TIFM_MMCSD_RXDE, sock->addr + SOCK_MMCSD_BUFFER_CONFIG); | ||
813 | |||
814 | // command timeout fixed to 64 clocks for now | ||
815 | writel(64, sock->addr + SOCK_MMCSD_COMMAND_TO); | ||
816 | writel(TIFM_MMCSD_INAB, sock->addr + SOCK_MMCSD_COMMAND); | ||
817 | |||
818 | /* INAB should take much less than reset */ | ||
819 | for (rc = 1; rc <= 16; rc <<= 1) { | ||
820 | host_status = readl(sock->addr + SOCK_MMCSD_STATUS); | ||
821 | writel(host_status, sock->addr + SOCK_MMCSD_STATUS); | ||
822 | if (!(host_status & TIFM_MMCSD_ERRMASK) | ||
823 | && (host_status & TIFM_MMCSD_EOC)) { | ||
824 | rc = 0; | ||
825 | break; | ||
826 | } | ||
827 | msleep(rc); | ||
828 | } | ||
829 | |||
830 | if (rc) { | ||
831 | printk(KERN_ERR DRIVER_NAME | ||
832 | ": card not ready - probe failed on initialization\n"); | ||
833 | return -ENODEV; | ||
834 | } | ||
835 | |||
836 | writel(TIFM_MMCSD_DATAMASK | TIFM_MMCSD_ERRMASK, | ||
837 | sock->addr + SOCK_MMCSD_INT_ENABLE); | ||
838 | mmiowb(); | ||
839 | |||
840 | return 0; | ||
779 | } | 841 | } |
780 | 842 | ||
781 | static int tifm_sd_probe(struct tifm_dev *sock) | 843 | static int tifm_sd_probe(struct tifm_dev *sock) |
@@ -784,8 +846,8 @@ static int tifm_sd_probe(struct tifm_dev *sock) | |||
784 | struct tifm_sd *host; | 846 | struct tifm_sd *host; |
785 | int rc = -EIO; | 847 | int rc = -EIO; |
786 | 848 | ||
787 | if (!(TIFM_SOCK_STATE_OCCUPIED & | 849 | if (!(TIFM_SOCK_STATE_OCCUPIED |
788 | readl(sock->addr + SOCK_PRESENT_STATE))) { | 850 | & readl(sock->addr + SOCK_PRESENT_STATE))) { |
789 | printk(KERN_WARNING DRIVER_NAME ": card gone, unexpectedly\n"); | 851 | printk(KERN_WARNING DRIVER_NAME ": card gone, unexpectedly\n"); |
790 | return rc; | 852 | return rc; |
791 | } | 853 | } |
@@ -795,109 +857,99 @@ static int tifm_sd_probe(struct tifm_dev *sock) | |||
795 | return -ENOMEM; | 857 | return -ENOMEM; |
796 | 858 | ||
797 | host = mmc_priv(mmc); | 859 | host = mmc_priv(mmc); |
798 | host->dev = sock; | ||
799 | host->clk_div = 61; | ||
800 | init_waitqueue_head(&host->can_eject); | ||
801 | INIT_WORK(&host->cmd_handler, tifm_sd_register_host); | ||
802 | INIT_DELAYED_WORK(&host->abort_handler, tifm_sd_abort); | ||
803 | |||
804 | tifm_set_drvdata(sock, mmc); | 860 | tifm_set_drvdata(sock, mmc); |
805 | sock->signal_irq = tifm_sd_signal_irq; | 861 | host->dev = sock; |
806 | |||
807 | host->clk_freq = 20000000; | ||
808 | host->timeout_jiffies = msecs_to_jiffies(1000); | 862 | host->timeout_jiffies = msecs_to_jiffies(1000); |
809 | 863 | ||
864 | init_waitqueue_head(&host->notify); | ||
865 | tasklet_init(&host->finish_tasklet, | ||
866 | no_dma ? tifm_sd_end_cmd_nodma : tifm_sd_end_cmd, | ||
867 | (unsigned long)host); | ||
868 | setup_timer(&host->timer, tifm_sd_abort, (unsigned long)host); | ||
869 | |||
810 | tifm_sd_ops.request = no_dma ? tifm_sd_request_nodma : tifm_sd_request; | 870 | tifm_sd_ops.request = no_dma ? tifm_sd_request_nodma : tifm_sd_request; |
811 | mmc->ops = &tifm_sd_ops; | 871 | mmc->ops = &tifm_sd_ops; |
812 | mmc->ocr_avail = MMC_VDD_32_33 | MMC_VDD_33_34; | 872 | mmc->ocr_avail = MMC_VDD_32_33 | MMC_VDD_33_34; |
813 | mmc->caps = MMC_CAP_4_BIT_DATA; | 873 | mmc->caps = MMC_CAP_4_BIT_DATA | MMC_CAP_MULTIWRITE; |
814 | mmc->f_min = 20000000 / 60; | 874 | mmc->f_min = 20000000 / 60; |
815 | mmc->f_max = 24000000; | 875 | mmc->f_max = 24000000; |
816 | mmc->max_hw_segs = 1; | 876 | mmc->max_hw_segs = 1; |
817 | mmc->max_phys_segs = 1; | 877 | mmc->max_phys_segs = 1; |
818 | mmc->max_sectors = 127; | 878 | // limited by DMA counter - it's safer to stick with |
819 | mmc->max_seg_size = mmc->max_sectors << 11; //2k maximum hw block length | 879 | // block counter has 11 bits though |
820 | 880 | mmc->max_blk_count = 256; | |
821 | writel(0, sock->addr + SOCK_MMCSD_INT_ENABLE); | 881 | // 2k maximum hw block length |
822 | writel(TIFM_MMCSD_RESET, sock->addr + SOCK_MMCSD_SYSTEM_CONTROL); | 882 | mmc->max_blk_size = 2048; |
823 | writel(host->clk_div | TIFM_MMCSD_POWER, | 883 | mmc->max_req_size = mmc->max_blk_size * mmc->max_blk_count; |
824 | sock->addr + SOCK_MMCSD_CONFIG); | 884 | mmc->max_seg_size = mmc->max_req_size; |
885 | sock->signal_irq = tifm_sd_signal_irq; | ||
886 | rc = tifm_sd_initialize_host(host); | ||
825 | 887 | ||
826 | for (rc = 0; rc < 50; rc++) { | 888 | if (!rc) |
827 | /* Wait for reset ack */ | 889 | rc = mmc_add_host(mmc); |
828 | if (1 & readl(sock->addr + SOCK_MMCSD_SYSTEM_STATUS)) { | 890 | if (rc) |
829 | rc = 0; | 891 | goto out_free_mmc; |
830 | break; | ||
831 | } | ||
832 | msleep(10); | ||
833 | } | ||
834 | 892 | ||
835 | if (rc) { | 893 | return 0; |
836 | printk(KERN_ERR DRIVER_NAME | 894 | out_free_mmc: |
837 | ": card not ready - probe failed\n"); | 895 | mmc_free_host(mmc); |
838 | mmc_free_host(mmc); | 896 | return rc; |
839 | return -ENODEV; | 897 | } |
840 | } | ||
841 | 898 | ||
842 | writel(0, sock->addr + SOCK_MMCSD_NUM_BLOCKS); | 899 | static void tifm_sd_remove(struct tifm_dev *sock) |
843 | writel(host->clk_div | TIFM_MMCSD_POWER, | 900 | { |
844 | sock->addr + SOCK_MMCSD_CONFIG); | 901 | struct mmc_host *mmc = tifm_get_drvdata(sock); |
845 | writel(TIFM_MMCSD_RXDE, sock->addr + SOCK_MMCSD_BUFFER_CONFIG); | 902 | struct tifm_sd *host = mmc_priv(mmc); |
846 | writel(TIFM_MMCSD_DATAMASK | TIFM_MMCSD_ERRMASK, | ||
847 | sock->addr + SOCK_MMCSD_INT_ENABLE); | ||
848 | 903 | ||
849 | writel(64, sock->addr + SOCK_MMCSD_COMMAND_TO); // command timeout 64 clocks for now | 904 | del_timer_sync(&host->timer); |
850 | writel(TIFM_MMCSD_INAB, sock->addr + SOCK_MMCSD_COMMAND); | 905 | tifm_sd_terminate(host); |
851 | writel(host->clk_div | TIFM_MMCSD_POWER, | 906 | wait_event_timeout(host->notify, host->flags & EJECT_DONE, |
852 | sock->addr + SOCK_MMCSD_CONFIG); | 907 | host->timeout_jiffies); |
908 | tasklet_kill(&host->finish_tasklet); | ||
909 | mmc_remove_host(mmc); | ||
853 | 910 | ||
854 | queue_delayed_work(sock->wq, &host->abort_handler, | 911 | /* The meaning of the bit majority in this constant is unknown. */ |
855 | host->timeout_jiffies); | 912 | writel(0xfff8 & readl(sock->addr + SOCK_CONTROL), |
913 | sock->addr + SOCK_CONTROL); | ||
856 | 914 | ||
857 | return 0; | 915 | tifm_set_drvdata(sock, NULL); |
916 | mmc_free_host(mmc); | ||
858 | } | 917 | } |
859 | 918 | ||
860 | static int tifm_sd_host_is_down(struct tifm_dev *sock) | 919 | #ifdef CONFIG_PM |
920 | |||
921 | static int tifm_sd_suspend(struct tifm_dev *sock, pm_message_t state) | ||
861 | { | 922 | { |
862 | struct mmc_host *mmc = tifm_get_drvdata(sock); | 923 | struct mmc_host *mmc = tifm_get_drvdata(sock); |
863 | struct tifm_sd *host = mmc_priv(mmc); | 924 | int rc; |
864 | unsigned long flags; | ||
865 | int rc = 0; | ||
866 | 925 | ||
867 | spin_lock_irqsave(&sock->lock, flags); | 926 | rc = mmc_suspend_host(mmc, state); |
868 | rc = (host->flags & EJECT_DONE); | 927 | /* The meaning of the bit majority in this constant is unknown. */ |
869 | spin_unlock_irqrestore(&sock->lock, flags); | 928 | writel(0xfff8 & readl(sock->addr + SOCK_CONTROL), |
929 | sock->addr + SOCK_CONTROL); | ||
870 | return rc; | 930 | return rc; |
871 | } | 931 | } |
872 | 932 | ||
873 | static void tifm_sd_remove(struct tifm_dev *sock) | 933 | static int tifm_sd_resume(struct tifm_dev *sock) |
874 | { | 934 | { |
875 | struct mmc_host *mmc = tifm_get_drvdata(sock); | 935 | struct mmc_host *mmc = tifm_get_drvdata(sock); |
876 | struct tifm_sd *host = mmc_priv(mmc); | 936 | struct tifm_sd *host = mmc_priv(mmc); |
877 | unsigned long flags; | ||
878 | 937 | ||
879 | spin_lock_irqsave(&sock->lock, flags); | 938 | if (sock->media_id != FM_SD |
880 | host->flags |= EJECT; | 939 | || tifm_sd_initialize_host(host)) { |
881 | if (host->req) | 940 | tifm_eject(sock); |
882 | queue_work(sock->wq, &host->cmd_handler); | 941 | return 0; |
883 | spin_unlock_irqrestore(&sock->lock, flags); | 942 | } else { |
884 | wait_event_timeout(host->can_eject, tifm_sd_host_is_down(sock), | 943 | return mmc_resume_host(mmc); |
885 | host->timeout_jiffies); | 944 | } |
945 | } | ||
886 | 946 | ||
887 | if (host->flags & HOST_REG) | 947 | #else |
888 | mmc_remove_host(mmc); | ||
889 | 948 | ||
890 | /* The meaning of the bit majority in this constant is unknown. */ | 949 | #define tifm_sd_suspend NULL |
891 | writel(0xfff8 & readl(sock->addr + SOCK_CONTROL), | 950 | #define tifm_sd_resume NULL |
892 | sock->addr + SOCK_CONTROL); | ||
893 | writel(0, sock->addr + SOCK_MMCSD_INT_ENABLE); | ||
894 | writel(TIFM_FIFO_INT_SETALL, | ||
895 | sock->addr + SOCK_DMA_FIFO_INT_ENABLE_CLEAR); | ||
896 | writel(0, sock->addr + SOCK_DMA_FIFO_INT_ENABLE_SET); | ||
897 | 951 | ||
898 | tifm_set_drvdata(sock, NULL); | 952 | #endif /* CONFIG_PM */ |
899 | mmc_free_host(mmc); | ||
900 | } | ||
901 | 953 | ||
902 | static tifm_media_id tifm_sd_id_tbl[] = { | 954 | static tifm_media_id tifm_sd_id_tbl[] = { |
903 | FM_SD, 0 | 955 | FM_SD, 0 |
@@ -910,7 +962,9 @@ static struct tifm_driver tifm_sd_driver = { | |||
910 | }, | 962 | }, |
911 | .id_table = tifm_sd_id_tbl, | 963 | .id_table = tifm_sd_id_tbl, |
912 | .probe = tifm_sd_probe, | 964 | .probe = tifm_sd_probe, |
913 | .remove = tifm_sd_remove | 965 | .remove = tifm_sd_remove, |
966 | .suspend = tifm_sd_suspend, | ||
967 | .resume = tifm_sd_resume | ||
914 | }; | 968 | }; |
915 | 969 | ||
916 | static int __init tifm_sd_init(void) | 970 | static int __init tifm_sd_init(void) |
diff --git a/drivers/mmc/wbsd.c b/drivers/mmc/wbsd.c index 7a282672f8e9..05ccfc43168f 100644 --- a/drivers/mmc/wbsd.c +++ b/drivers/mmc/wbsd.c | |||
@@ -1,7 +1,7 @@ | |||
1 | /* | 1 | /* |
2 | * linux/drivers/mmc/wbsd.c - Winbond W83L51xD SD/MMC driver | 2 | * linux/drivers/mmc/wbsd.c - Winbond W83L51xD SD/MMC driver |
3 | * | 3 | * |
4 | * Copyright (C) 2004-2005 Pierre Ossman, All Rights Reserved. | 4 | * Copyright (C) 2004-2006 Pierre Ossman, All Rights Reserved. |
5 | * | 5 | * |
6 | * This program is free software; you can redistribute it and/or modify | 6 | * This program is free software; you can redistribute it and/or modify |
7 | * it under the terms of the GNU General Public License as published by | 7 | * it under the terms of the GNU General Public License as published by |
@@ -42,7 +42,6 @@ | |||
42 | #include "wbsd.h" | 42 | #include "wbsd.h" |
43 | 43 | ||
44 | #define DRIVER_NAME "wbsd" | 44 | #define DRIVER_NAME "wbsd" |
45 | #define DRIVER_VERSION "1.6" | ||
46 | 45 | ||
47 | #define DBG(x...) \ | 46 | #define DBG(x...) \ |
48 | pr_debug(DRIVER_NAME ": " x) | 47 | pr_debug(DRIVER_NAME ": " x) |
@@ -272,16 +271,9 @@ static inline int wbsd_next_sg(struct wbsd_host *host) | |||
272 | return host->num_sg; | 271 | return host->num_sg; |
273 | } | 272 | } |
274 | 273 | ||
275 | static inline char *wbsd_kmap_sg(struct wbsd_host *host) | 274 | static inline char *wbsd_sg_to_buffer(struct wbsd_host *host) |
276 | { | 275 | { |
277 | host->mapped_sg = kmap_atomic(host->cur_sg->page, KM_BIO_SRC_IRQ) + | 276 | return page_address(host->cur_sg->page) + host->cur_sg->offset; |
278 | host->cur_sg->offset; | ||
279 | return host->mapped_sg; | ||
280 | } | ||
281 | |||
282 | static inline void wbsd_kunmap_sg(struct wbsd_host *host) | ||
283 | { | ||
284 | kunmap_atomic(host->mapped_sg, KM_BIO_SRC_IRQ); | ||
285 | } | 277 | } |
286 | 278 | ||
287 | static inline void wbsd_sg_to_dma(struct wbsd_host *host, struct mmc_data *data) | 279 | static inline void wbsd_sg_to_dma(struct wbsd_host *host, struct mmc_data *data) |
@@ -302,12 +294,11 @@ static inline void wbsd_sg_to_dma(struct wbsd_host *host, struct mmc_data *data) | |||
302 | * we do not transfer too much. | 294 | * we do not transfer too much. |
303 | */ | 295 | */ |
304 | for (i = 0; i < len; i++) { | 296 | for (i = 0; i < len; i++) { |
305 | sgbuf = kmap_atomic(sg[i].page, KM_BIO_SRC_IRQ) + sg[i].offset; | 297 | sgbuf = page_address(sg[i].page) + sg[i].offset; |
306 | if (size < sg[i].length) | 298 | if (size < sg[i].length) |
307 | memcpy(dmabuf, sgbuf, size); | 299 | memcpy(dmabuf, sgbuf, size); |
308 | else | 300 | else |
309 | memcpy(dmabuf, sgbuf, sg[i].length); | 301 | memcpy(dmabuf, sgbuf, sg[i].length); |
310 | kunmap_atomic(sgbuf, KM_BIO_SRC_IRQ); | ||
311 | dmabuf += sg[i].length; | 302 | dmabuf += sg[i].length; |
312 | 303 | ||
313 | if (size < sg[i].length) | 304 | if (size < sg[i].length) |
@@ -347,12 +338,11 @@ static inline void wbsd_dma_to_sg(struct wbsd_host *host, struct mmc_data *data) | |||
347 | * we do not transfer too much. | 338 | * we do not transfer too much. |
348 | */ | 339 | */ |
349 | for (i = 0; i < len; i++) { | 340 | for (i = 0; i < len; i++) { |
350 | sgbuf = kmap_atomic(sg[i].page, KM_BIO_SRC_IRQ) + sg[i].offset; | 341 | sgbuf = page_address(sg[i].page) + sg[i].offset; |
351 | if (size < sg[i].length) | 342 | if (size < sg[i].length) |
352 | memcpy(sgbuf, dmabuf, size); | 343 | memcpy(sgbuf, dmabuf, size); |
353 | else | 344 | else |
354 | memcpy(sgbuf, dmabuf, sg[i].length); | 345 | memcpy(sgbuf, dmabuf, sg[i].length); |
355 | kunmap_atomic(sgbuf, KM_BIO_SRC_IRQ); | ||
356 | dmabuf += sg[i].length; | 346 | dmabuf += sg[i].length; |
357 | 347 | ||
358 | if (size < sg[i].length) | 348 | if (size < sg[i].length) |
@@ -497,7 +487,7 @@ static void wbsd_empty_fifo(struct wbsd_host *host) | |||
497 | if (data->bytes_xfered == host->size) | 487 | if (data->bytes_xfered == host->size) |
498 | return; | 488 | return; |
499 | 489 | ||
500 | buffer = wbsd_kmap_sg(host) + host->offset; | 490 | buffer = wbsd_sg_to_buffer(host) + host->offset; |
501 | 491 | ||
502 | /* | 492 | /* |
503 | * Drain the fifo. This has a tendency to loop longer | 493 | * Drain the fifo. This has a tendency to loop longer |
@@ -526,17 +516,13 @@ static void wbsd_empty_fifo(struct wbsd_host *host) | |||
526 | /* | 516 | /* |
527 | * Transfer done? | 517 | * Transfer done? |
528 | */ | 518 | */ |
529 | if (data->bytes_xfered == host->size) { | 519 | if (data->bytes_xfered == host->size) |
530 | wbsd_kunmap_sg(host); | ||
531 | return; | 520 | return; |
532 | } | ||
533 | 521 | ||
534 | /* | 522 | /* |
535 | * End of scatter list entry? | 523 | * End of scatter list entry? |
536 | */ | 524 | */ |
537 | if (host->remain == 0) { | 525 | if (host->remain == 0) { |
538 | wbsd_kunmap_sg(host); | ||
539 | |||
540 | /* | 526 | /* |
541 | * Get next entry. Check if last. | 527 | * Get next entry. Check if last. |
542 | */ | 528 | */ |
@@ -554,13 +540,11 @@ static void wbsd_empty_fifo(struct wbsd_host *host) | |||
554 | return; | 540 | return; |
555 | } | 541 | } |
556 | 542 | ||
557 | buffer = wbsd_kmap_sg(host); | 543 | buffer = wbsd_sg_to_buffer(host); |
558 | } | 544 | } |
559 | } | 545 | } |
560 | } | 546 | } |
561 | 547 | ||
562 | wbsd_kunmap_sg(host); | ||
563 | |||
564 | /* | 548 | /* |
565 | * This is a very dirty hack to solve a | 549 | * This is a very dirty hack to solve a |
566 | * hardware problem. The chip doesn't trigger | 550 | * hardware problem. The chip doesn't trigger |
@@ -583,7 +567,7 @@ static void wbsd_fill_fifo(struct wbsd_host *host) | |||
583 | if (data->bytes_xfered == host->size) | 567 | if (data->bytes_xfered == host->size) |
584 | return; | 568 | return; |
585 | 569 | ||
586 | buffer = wbsd_kmap_sg(host) + host->offset; | 570 | buffer = wbsd_sg_to_buffer(host) + host->offset; |
587 | 571 | ||
588 | /* | 572 | /* |
589 | * Fill the fifo. This has a tendency to loop longer | 573 | * Fill the fifo. This has a tendency to loop longer |
@@ -612,17 +596,13 @@ static void wbsd_fill_fifo(struct wbsd_host *host) | |||
612 | /* | 596 | /* |
613 | * Transfer done? | 597 | * Transfer done? |
614 | */ | 598 | */ |
615 | if (data->bytes_xfered == host->size) { | 599 | if (data->bytes_xfered == host->size) |
616 | wbsd_kunmap_sg(host); | ||
617 | return; | 600 | return; |
618 | } | ||
619 | 601 | ||
620 | /* | 602 | /* |
621 | * End of scatter list entry? | 603 | * End of scatter list entry? |
622 | */ | 604 | */ |
623 | if (host->remain == 0) { | 605 | if (host->remain == 0) { |
624 | wbsd_kunmap_sg(host); | ||
625 | |||
626 | /* | 606 | /* |
627 | * Get next entry. Check if last. | 607 | * Get next entry. Check if last. |
628 | */ | 608 | */ |
@@ -640,13 +620,11 @@ static void wbsd_fill_fifo(struct wbsd_host *host) | |||
640 | return; | 620 | return; |
641 | } | 621 | } |
642 | 622 | ||
643 | buffer = wbsd_kmap_sg(host); | 623 | buffer = wbsd_sg_to_buffer(host); |
644 | } | 624 | } |
645 | } | 625 | } |
646 | } | 626 | } |
647 | 627 | ||
648 | wbsd_kunmap_sg(host); | ||
649 | |||
650 | /* | 628 | /* |
651 | * The controller stops sending interrupts for | 629 | * The controller stops sending interrupts for |
652 | * 'FIFO empty' under certain conditions. So we | 630 | * 'FIFO empty' under certain conditions. So we |
@@ -910,6 +888,45 @@ static void wbsd_request(struct mmc_host *mmc, struct mmc_request *mrq) | |||
910 | */ | 888 | */ |
911 | if (cmd->data && (cmd->error == MMC_ERR_NONE)) { | 889 | if (cmd->data && (cmd->error == MMC_ERR_NONE)) { |
912 | /* | 890 | /* |
891 | * The hardware is so delightfully stupid that it has a list | ||
892 | * of "data" commands. If a command isn't on this list, it'll | ||
893 | * just go back to the idle state and won't send any data | ||
894 | * interrupts. | ||
895 | */ | ||
896 | switch (cmd->opcode) { | ||
897 | case 11: | ||
898 | case 17: | ||
899 | case 18: | ||
900 | case 20: | ||
901 | case 24: | ||
902 | case 25: | ||
903 | case 26: | ||
904 | case 27: | ||
905 | case 30: | ||
906 | case 42: | ||
907 | case 56: | ||
908 | break; | ||
909 | |||
910 | /* ACMDs. We don't keep track of state, so we just treat them | ||
911 | * like any other command. */ | ||
912 | case 51: | ||
913 | break; | ||
914 | |||
915 | default: | ||
916 | #ifdef CONFIG_MMC_DEBUG | ||
917 | printk(KERN_WARNING "%s: Data command %d is not " | ||
918 | "supported by this controller.\n", | ||
919 | mmc_hostname(host->mmc), cmd->opcode); | ||
920 | #endif | ||
921 | cmd->data->error = MMC_ERR_INVALID; | ||
922 | |||
923 | if (cmd->data->stop) | ||
924 | wbsd_send_command(host, cmd->data->stop); | ||
925 | |||
926 | goto done; | ||
927 | }; | ||
928 | |||
929 | /* | ||
913 | * Dirty fix for hardware bug. | 930 | * Dirty fix for hardware bug. |
914 | */ | 931 | */ |
915 | if (host->dma == -1) | 932 | if (host->dma == -1) |
@@ -1343,16 +1360,27 @@ static int __devinit wbsd_alloc_mmc(struct device *dev) | |||
1343 | mmc->max_phys_segs = 128; | 1360 | mmc->max_phys_segs = 128; |
1344 | 1361 | ||
1345 | /* | 1362 | /* |
1346 | * Maximum number of sectors in one transfer. Also limited by 64kB | 1363 | * Maximum request size. Also limited by 64KiB buffer. |
1347 | * buffer. | ||
1348 | */ | 1364 | */ |
1349 | mmc->max_sectors = 128; | 1365 | mmc->max_req_size = 65536; |
1350 | 1366 | ||
1351 | /* | 1367 | /* |
1352 | * Maximum segment size. Could be one segment with the maximum number | 1368 | * Maximum segment size. Could be one segment with the maximum number |
1353 | * of segments. | 1369 | * of bytes. |
1370 | */ | ||
1371 | mmc->max_seg_size = mmc->max_req_size; | ||
1372 | |||
1373 | /* | ||
1374 | * Maximum block size. We have 12 bits (= 4095) but have to subtract | ||
1375 | * space for CRC. So the maximum is 4095 - 4*2 = 4087. | ||
1376 | */ | ||
1377 | mmc->max_blk_size = 4087; | ||
1378 | |||
1379 | /* | ||
1380 | * Maximum block count. There is no real limit so the maximum | ||
1381 | * request size will be the only restriction. | ||
1354 | */ | 1382 | */ |
1355 | mmc->max_seg_size = mmc->max_sectors * 512; | 1383 | mmc->max_blk_count = mmc->max_req_size; |
1356 | 1384 | ||
1357 | dev_set_drvdata(dev, mmc); | 1385 | dev_set_drvdata(dev, mmc); |
1358 | 1386 | ||
@@ -2071,8 +2099,7 @@ static int __init wbsd_drv_init(void) | |||
2071 | int result; | 2099 | int result; |
2072 | 2100 | ||
2073 | printk(KERN_INFO DRIVER_NAME | 2101 | printk(KERN_INFO DRIVER_NAME |
2074 | ": Winbond W83L51xD SD/MMC card interface driver, " | 2102 | ": Winbond W83L51xD SD/MMC card interface driver\n"); |
2075 | DRIVER_VERSION "\n"); | ||
2076 | printk(KERN_INFO DRIVER_NAME ": Copyright(c) Pierre Ossman\n"); | 2103 | printk(KERN_INFO DRIVER_NAME ": Copyright(c) Pierre Ossman\n"); |
2077 | 2104 | ||
2078 | #ifdef CONFIG_PNP | 2105 | #ifdef CONFIG_PNP |
@@ -2136,7 +2163,6 @@ module_param(dma, int, 0444); | |||
2136 | MODULE_LICENSE("GPL"); | 2163 | MODULE_LICENSE("GPL"); |
2137 | MODULE_AUTHOR("Pierre Ossman <drzeus@drzeus.cx>"); | 2164 | MODULE_AUTHOR("Pierre Ossman <drzeus@drzeus.cx>"); |
2138 | MODULE_DESCRIPTION("Winbond W83L51xD SD/MMC card interface driver"); | 2165 | MODULE_DESCRIPTION("Winbond W83L51xD SD/MMC card interface driver"); |
2139 | MODULE_VERSION(DRIVER_VERSION); | ||
2140 | 2166 | ||
2141 | #ifdef CONFIG_PNP | 2167 | #ifdef CONFIG_PNP |
2142 | MODULE_PARM_DESC(nopnp, "Scan for device instead of relying on PNP. (default 0)"); | 2168 | MODULE_PARM_DESC(nopnp, "Scan for device instead of relying on PNP. (default 0)"); |
diff --git a/drivers/mmc/wbsd.h b/drivers/mmc/wbsd.h index 6072993f01e3..d06718b0e2ab 100644 --- a/drivers/mmc/wbsd.h +++ b/drivers/mmc/wbsd.h | |||
@@ -154,7 +154,6 @@ struct wbsd_host | |||
154 | 154 | ||
155 | struct scatterlist* cur_sg; /* Current SG entry */ | 155 | struct scatterlist* cur_sg; /* Current SG entry */ |
156 | unsigned int num_sg; /* Number of entries left */ | 156 | unsigned int num_sg; /* Number of entries left */ |
157 | void* mapped_sg; /* vaddr of mapped sg */ | ||
158 | 157 | ||
159 | unsigned int offset; /* Offset into current entry */ | 158 | unsigned int offset; /* Offset into current entry */ |
160 | unsigned int remain; /* Data left in curren entry */ | 159 | unsigned int remain; /* Data left in curren entry */ |