Lines Matching +full:mmc +full:- +full:slot
1 // SPDX-License-Identifier: GPL-2.0-only
3 * cb710/mmc.c
5 * Copyright by Michał Mirosław, 2008-2009
11 #include "cb710-mmc.h"
20 (ARRAY_SIZE(cb710_clock_divider_log2) - 1)
27 static void cb710_mmc_select_clock_divider(struct mmc_host *mmc, int hz) in cb710_mmc_select_clock_divider() argument
29 struct cb710_slot *slot = cb710_mmc_to_slot(mmc); in cb710_mmc_select_clock_divider() local
30 struct pci_dev *pdev = cb710_slot_to_chip(slot)->pdev; in cb710_mmc_select_clock_divider()
37 * indexes 1-7 work as written in the table in cb710_mmc_select_clock_divider()
38 * indexes 0,8-15 give no clock output in cb710_mmc_select_clock_divider()
56 dev_dbg(cb710_slot_dev(slot), in cb710_mmc_select_clock_divider()
62 static void __cb710_mmc_enable_irq(struct cb710_slot *slot, in __cb710_mmc_enable_irq() argument
66 * - it gets set later if any interrupt sources are enabled */ in __cb710_mmc_enable_irq()
71 * -> bit 15 port 0x0C seems to be global interrupt enable in __cb710_mmc_enable_irq()
74 enable = (cb710_read_port_16(slot, CB710_MMC_IRQ_ENABLE_PORT) in __cb710_mmc_enable_irq()
80 cb710_write_port_16(slot, CB710_MMC_IRQ_ENABLE_PORT, enable); in __cb710_mmc_enable_irq()
83 static void cb710_mmc_enable_irq(struct cb710_slot *slot, in cb710_mmc_enable_irq() argument
86 struct cb710_mmc_reader *reader = mmc_priv(cb710_slot_to_mmc(slot)); in cb710_mmc_enable_irq()
89 spin_lock_irqsave(&reader->irq_lock, flags); in cb710_mmc_enable_irq()
91 __cb710_mmc_enable_irq(slot, enable, mask); in cb710_mmc_enable_irq()
92 spin_unlock_irqrestore(&reader->irq_lock, flags); in cb710_mmc_enable_irq()
95 static void cb710_mmc_reset_events(struct cb710_slot *slot) in cb710_mmc_reset_events() argument
97 cb710_write_port_8(slot, CB710_MMC_STATUS0_PORT, 0xFF); in cb710_mmc_reset_events()
98 cb710_write_port_8(slot, CB710_MMC_STATUS1_PORT, 0xFF); in cb710_mmc_reset_events()
99 cb710_write_port_8(slot, CB710_MMC_STATUS2_PORT, 0xFF); in cb710_mmc_reset_events()
102 static void cb710_mmc_enable_4bit_data(struct cb710_slot *slot, int enable) in cb710_mmc_enable_4bit_data() argument
105 cb710_modify_port_8(slot, CB710_MMC_CONFIG1_PORT, in cb710_mmc_enable_4bit_data()
108 cb710_modify_port_8(slot, CB710_MMC_CONFIG1_PORT, in cb710_mmc_enable_4bit_data()
112 static int cb710_check_event(struct cb710_slot *slot, u8 what) in cb710_check_event() argument
116 status = cb710_read_port_16(slot, CB710_MMC_STATUS_PORT); in cb710_check_event()
120 dev_dbg(cb710_slot_dev(slot), in cb710_check_event()
122 cb710_write_port_8(slot, CB710_MMC_STATUS0_PORT, in cb710_check_event()
128 dev_dbg(cb710_slot_dev(slot), in cb710_check_event()
130 cb710_write_port_8(slot, CB710_MMC_STATUS0_PORT, status & 0xFF); in cb710_check_event()
131 cb710_write_port_8(slot, CB710_MMC_STATUS1_PORT, in cb710_check_event()
133 return -EIO; in cb710_check_event()
138 cb710_write_port_8(slot, CB710_MMC_STATUS1_PORT, what); in cb710_check_event()
145 static int cb710_wait_for_event(struct cb710_slot *slot, u8 what) in cb710_wait_for_event() argument
152 e = cb710_read_port_32(slot, CB710_MMC_STATUS_PORT); in cb710_wait_for_event()
155 while (!(err = cb710_check_event(slot, what))) { in cb710_wait_for_event()
156 if (!--limit) { in cb710_wait_for_event()
157 cb710_dump_regs(cb710_slot_to_chip(slot), in cb710_wait_for_event()
159 err = -ETIMEDOUT; in cb710_wait_for_event()
166 x = cb710_read_port_32(slot, CB710_MMC_STATUS_PORT); in cb710_wait_for_event()
168 limit = 2000000 - limit; in cb710_wait_for_event()
170 dev_dbg(cb710_slot_dev(slot), in cb710_wait_for_event()
178 static int cb710_wait_while_busy(struct cb710_slot *slot, uint8_t mask) in cb710_wait_while_busy() argument
185 e = cb710_read_port_32(slot, CB710_MMC_STATUS_PORT); in cb710_wait_while_busy()
188 while (cb710_read_port_8(slot, CB710_MMC_STATUS2_PORT) & mask) { in cb710_wait_while_busy()
189 if (!--limit) { in cb710_wait_while_busy()
190 cb710_dump_regs(cb710_slot_to_chip(slot), in cb710_wait_while_busy()
192 err = -ETIMEDOUT; in cb710_wait_while_busy()
199 x = cb710_read_port_32(slot, CB710_MMC_STATUS_PORT); in cb710_wait_while_busy()
201 limit = 500000 - limit; in cb710_wait_while_busy()
203 dev_dbg(cb710_slot_dev(slot), in cb710_wait_while_busy()
210 static void cb710_mmc_set_transfer_size(struct cb710_slot *slot, in cb710_mmc_set_transfer_size() argument
213 cb710_wait_while_busy(slot, CB710_MMC_S2_BUSY_20); in cb710_mmc_set_transfer_size()
214 cb710_write_port_32(slot, CB710_MMC_TRANSFER_SIZE_PORT, in cb710_mmc_set_transfer_size()
215 ((count - 1) << 16)|(blocksize - 1)); in cb710_mmc_set_transfer_size()
217 dev_vdbg(cb710_slot_dev(slot), "set up for %zu block%s of %zu bytes\n", in cb710_mmc_set_transfer_size()
221 static void cb710_mmc_fifo_hack(struct cb710_slot *slot) in cb710_mmc_fifo_hack() argument
223 /* without this, received data is prepended with 8-bytes of zeroes */ in cb710_mmc_fifo_hack()
227 r1 = cb710_read_port_32(slot, CB710_MMC_DATA_PORT); in cb710_mmc_fifo_hack()
228 r2 = cb710_read_port_32(slot, CB710_MMC_DATA_PORT); in cb710_mmc_fifo_hack()
229 if (cb710_read_port_8(slot, CB710_MMC_STATUS0_PORT) in cb710_mmc_fifo_hack()
231 cb710_write_port_8(slot, CB710_MMC_STATUS0_PORT, in cb710_mmc_fifo_hack()
236 dev_dbg(cb710_slot_dev(slot), in cb710_mmc_fifo_hack()
237 "FIFO-read-hack: expected STATUS0 bit was %s\n", in cb710_mmc_fifo_hack()
239 dev_dbg(cb710_slot_dev(slot), in cb710_mmc_fifo_hack()
240 "FIFO-read-hack: dwords ignored: %08X %08X - %s\n", in cb710_mmc_fifo_hack()
244 static int cb710_mmc_receive_pio(struct cb710_slot *slot, in cb710_mmc_receive_pio() argument
247 if (!(cb710_read_port_8(slot, CB710_MMC_STATUS2_PORT) & CB710_MMC_S2_FIFO_READY)) { in cb710_mmc_receive_pio()
248 int err = cb710_wait_for_event(slot, in cb710_mmc_receive_pio()
255 slot->iobase + CB710_MMC_DATA_PORT, dw_count); in cb710_mmc_receive_pio()
262 return !(data->blksz & 15 && (data->blocks != 1 || data->blksz != 8)); in cb710_is_transfer_size_supported()
265 static int cb710_mmc_receive(struct cb710_slot *slot, struct mmc_data *data) in cb710_mmc_receive() argument
268 size_t len, blocks = data->blocks; in cb710_mmc_receive()
271 /* TODO: I don't know how/if the hardware handles non-16B-boundary blocks in cb710_mmc_receive()
273 if (unlikely(data->blksz & 15 && (data->blocks != 1 || data->blksz != 8))) in cb710_mmc_receive()
274 return -EINVAL; in cb710_mmc_receive()
276 sg_miter_start(&miter, data->sg, data->sg_len, SG_MITER_TO_SG); in cb710_mmc_receive()
278 cb710_modify_port_8(slot, CB710_MMC_CONFIG2_PORT, in cb710_mmc_receive()
281 cb710_mmc_fifo_hack(slot); in cb710_mmc_receive()
283 while (blocks-- > 0) { in cb710_mmc_receive()
284 len = data->blksz; in cb710_mmc_receive()
287 err = cb710_mmc_receive_pio(slot, &miter, 4); in cb710_mmc_receive()
290 len -= 16; in cb710_mmc_receive()
296 cb710_modify_port_8(slot, CB710_MMC_CONFIG2_PORT, in cb710_mmc_receive()
297 len - 1, CB710_MMC_C2_READ_PIO_SIZE_MASK); in cb710_mmc_receive()
300 err = cb710_mmc_receive_pio(slot, &miter, len); in cb710_mmc_receive()
309 static int cb710_mmc_send(struct cb710_slot *slot, struct mmc_data *data) in cb710_mmc_send() argument
312 size_t len, blocks = data->blocks; in cb710_mmc_send()
316 * non-16B-boundary blocks */ in cb710_mmc_send()
317 if (unlikely(data->blocks > 1 && data->blksz & 15)) in cb710_mmc_send()
318 return -EINVAL; in cb710_mmc_send()
320 sg_miter_start(&miter, data->sg, data->sg_len, SG_MITER_FROM_SG); in cb710_mmc_send()
322 cb710_modify_port_8(slot, CB710_MMC_CONFIG2_PORT, in cb710_mmc_send()
325 while (blocks-- > 0) { in cb710_mmc_send()
326 len = (data->blksz + 15) >> 4; in cb710_mmc_send()
328 if (!(cb710_read_port_8(slot, CB710_MMC_STATUS2_PORT) in cb710_mmc_send()
330 err = cb710_wait_for_event(slot, in cb710_mmc_send()
336 slot->iobase + CB710_MMC_DATA_PORT, 4); in cb710_mmc_send()
337 } while (--len); in cb710_mmc_send()
347 unsigned int flags = cmd->flags; in cb710_encode_cmd_flags()
355 * Original driver set bit 14 for MMC/SD application in cb710_encode_cmd_flags()
370 cb_flags |= cmd->opcode << CB710_MMC_CMD_CODE_SHIFT; in cb710_encode_cmd_flags()
372 if (cmd->data && (cmd->data->flags & MMC_DATA_READ)) in cb710_encode_cmd_flags()
395 static void cb710_receive_response(struct cb710_slot *slot, in cb710_receive_response() argument
401 if (cmd->flags & MMC_RSP_136) { in cb710_receive_response()
404 resp[0] = cb710_read_port_32(slot, CB710_MMC_RESPONSE3_PORT); in cb710_receive_response()
405 resp[1] = cb710_read_port_32(slot, CB710_MMC_RESPONSE2_PORT); in cb710_receive_response()
406 resp[2] = cb710_read_port_32(slot, CB710_MMC_RESPONSE1_PORT); in cb710_receive_response()
407 resp[3] = cb710_read_port_32(slot, CB710_MMC_RESPONSE0_PORT); in cb710_receive_response()
410 cmd->resp[0] = (resp[0] << 8)|(resp[1] >> 24); in cb710_receive_response()
411 cmd->resp[1] = (resp[1] << 8)|(resp[2] >> 24); in cb710_receive_response()
412 cmd->resp[2] = (resp[2] << 8)|(resp[3] >> 24); in cb710_receive_response()
413 cmd->resp[3] = (resp[3] << 8); in cb710_receive_response()
415 rsp_opcode = cb710_read_port_32(slot, CB710_MMC_RESPONSE1_PORT) & 0x3F; in cb710_receive_response()
416 cmd->resp[0] = cb710_read_port_32(slot, CB710_MMC_RESPONSE0_PORT); in cb710_receive_response()
419 wanted_opcode = (cmd->flags & MMC_RSP_OPCODE) ? cmd->opcode : 0x3F; in cb710_receive_response()
421 cmd->error = -EILSEQ; in cb710_receive_response()
424 static int cb710_mmc_transfer_data(struct cb710_slot *slot, in cb710_mmc_transfer_data() argument
429 if (data->flags & MMC_DATA_READ) in cb710_mmc_transfer_data()
430 error = cb710_mmc_receive(slot, data); in cb710_mmc_transfer_data()
432 error = cb710_mmc_send(slot, data); in cb710_mmc_transfer_data()
434 to = cb710_wait_for_event(slot, CB710_MMC_S1_DATA_TRANSFER_DONE); in cb710_mmc_transfer_data()
439 data->bytes_xfered = data->blksz * data->blocks; in cb710_mmc_transfer_data()
443 static int cb710_mmc_command(struct mmc_host *mmc, struct mmc_command *cmd) in cb710_mmc_command() argument
445 struct cb710_slot *slot = cb710_mmc_to_slot(mmc); in cb710_mmc_command() local
446 struct cb710_mmc_reader *reader = mmc_priv(mmc); in cb710_mmc_command()
447 struct mmc_data *data = cmd->data; in cb710_mmc_command()
450 dev_dbg(cb710_slot_dev(slot), "cmd request: 0x%04X\n", cb_cmd); in cb710_mmc_command()
454 data->error = -EINVAL; in cb710_mmc_command()
455 return -1; in cb710_mmc_command()
457 cb710_mmc_set_transfer_size(slot, data->blocks, data->blksz); in cb710_mmc_command()
460 cb710_wait_while_busy(slot, CB710_MMC_S2_BUSY_20|CB710_MMC_S2_BUSY_10); in cb710_mmc_command()
461 cb710_write_port_16(slot, CB710_MMC_CMD_TYPE_PORT, cb_cmd); in cb710_mmc_command()
462 cb710_wait_while_busy(slot, CB710_MMC_S2_BUSY_20); in cb710_mmc_command()
463 cb710_write_port_32(slot, CB710_MMC_CMD_PARAM_PORT, cmd->arg); in cb710_mmc_command()
464 cb710_mmc_reset_events(slot); in cb710_mmc_command()
465 cb710_wait_while_busy(slot, CB710_MMC_S2_BUSY_20); in cb710_mmc_command()
466 cb710_modify_port_8(slot, CB710_MMC_CONFIG0_PORT, 0x01, 0); in cb710_mmc_command()
468 cmd->error = cb710_wait_for_event(slot, CB710_MMC_S1_COMMAND_SENT); in cb710_mmc_command()
469 if (cmd->error) in cb710_mmc_command()
470 return -1; in cb710_mmc_command()
472 if (cmd->flags & MMC_RSP_PRESENT) { in cb710_mmc_command()
473 cb710_receive_response(slot, cmd); in cb710_mmc_command()
474 if (cmd->error) in cb710_mmc_command()
475 return -1; in cb710_mmc_command()
479 data->error = cb710_mmc_transfer_data(slot, data); in cb710_mmc_command()
483 static void cb710_mmc_request(struct mmc_host *mmc, struct mmc_request *mrq) in cb710_mmc_request() argument
485 struct cb710_slot *slot = cb710_mmc_to_slot(mmc); in cb710_mmc_request() local
486 struct cb710_mmc_reader *reader = mmc_priv(mmc); in cb710_mmc_request()
488 WARN_ON(reader->mrq != NULL); in cb710_mmc_request()
490 reader->mrq = mrq; in cb710_mmc_request()
491 cb710_mmc_enable_irq(slot, CB710_MMC_IE_TEST_MASK, 0); in cb710_mmc_request()
493 if (!cb710_mmc_command(mmc, mrq->cmd) && mrq->stop) in cb710_mmc_request()
494 cb710_mmc_command(mmc, mrq->stop); in cb710_mmc_request()
496 queue_work(system_bh_wq, &reader->finish_req_bh_work); in cb710_mmc_request()
499 static int cb710_mmc_powerup(struct cb710_slot *slot) in cb710_mmc_powerup() argument
502 struct cb710_chip *chip = cb710_slot_to_chip(slot); in cb710_mmc_powerup()
507 dev_dbg(cb710_slot_dev(slot), "bus powerup\n"); in cb710_mmc_powerup()
509 err = cb710_wait_while_busy(slot, CB710_MMC_S2_BUSY_20); in cb710_mmc_powerup()
512 cb710_modify_port_8(slot, CB710_MMC_CONFIG1_PORT, 0x80, 0); in cb710_mmc_powerup()
513 cb710_modify_port_8(slot, CB710_MMC_CONFIG3_PORT, 0x80, 0); in cb710_mmc_powerup()
516 dev_dbg(cb710_slot_dev(slot), "after delay 1\n"); in cb710_mmc_powerup()
518 err = cb710_wait_while_busy(slot, CB710_MMC_S2_BUSY_20); in cb710_mmc_powerup()
521 cb710_modify_port_8(slot, CB710_MMC_CONFIG1_PORT, 0x09, 0); in cb710_mmc_powerup()
524 dev_dbg(cb710_slot_dev(slot), "after delay 2\n"); in cb710_mmc_powerup()
526 err = cb710_wait_while_busy(slot, CB710_MMC_S2_BUSY_20); in cb710_mmc_powerup()
529 cb710_modify_port_8(slot, CB710_MMC_CONFIG1_PORT, 0, 0x08); in cb710_mmc_powerup()
532 dev_dbg(cb710_slot_dev(slot), "after delay 3\n"); in cb710_mmc_powerup()
534 cb710_modify_port_8(slot, CB710_MMC_CONFIG0_PORT, 0x06, 0); in cb710_mmc_powerup()
535 cb710_modify_port_8(slot, CB710_MMC_CONFIG1_PORT, 0x70, 0); in cb710_mmc_powerup()
536 cb710_modify_port_8(slot, CB710_MMC_CONFIG2_PORT, 0x80, 0); in cb710_mmc_powerup()
537 cb710_modify_port_8(slot, CB710_MMC_CONFIG3_PORT, 0x03, 0); in cb710_mmc_powerup()
539 err = cb710_wait_while_busy(slot, CB710_MMC_S2_BUSY_20); in cb710_mmc_powerup()
545 * (it doesn't depend on write-to-read delay) */ in cb710_mmc_powerup()
546 cb710_write_port_16(slot, CB710_MMC_CONFIGB_PORT, 0xFFFF); in cb710_mmc_powerup()
547 cb710_modify_port_8(slot, CB710_MMC_CONFIG0_PORT, 0x06, 0); in cb710_mmc_powerup()
549 dev_dbg(cb710_slot_dev(slot), "bus powerup finished\n"); in cb710_mmc_powerup()
551 return cb710_check_event(slot, 0); in cb710_mmc_powerup()
554 static void cb710_mmc_powerdown(struct cb710_slot *slot) in cb710_mmc_powerdown() argument
556 cb710_modify_port_8(slot, CB710_MMC_CONFIG1_PORT, 0, 0x81); in cb710_mmc_powerdown()
557 cb710_modify_port_8(slot, CB710_MMC_CONFIG3_PORT, 0, 0x80); in cb710_mmc_powerdown()
560 static void cb710_mmc_set_ios(struct mmc_host *mmc, struct mmc_ios *ios) in cb710_mmc_set_ios() argument
562 struct cb710_slot *slot = cb710_mmc_to_slot(mmc); in cb710_mmc_set_ios() local
563 struct cb710_mmc_reader *reader = mmc_priv(mmc); in cb710_mmc_set_ios()
566 cb710_mmc_select_clock_divider(mmc, ios->clock); in cb710_mmc_set_ios()
568 if (ios->power_mode != reader->last_power_mode) { in cb710_mmc_set_ios()
569 switch (ios->power_mode) { in cb710_mmc_set_ios()
571 err = cb710_mmc_powerup(slot); in cb710_mmc_set_ios()
573 dev_warn(cb710_slot_dev(slot), in cb710_mmc_set_ios()
574 "powerup failed (%d)- retrying\n", err); in cb710_mmc_set_ios()
575 cb710_mmc_powerdown(slot); in cb710_mmc_set_ios()
577 err = cb710_mmc_powerup(slot); in cb710_mmc_set_ios()
579 dev_warn(cb710_slot_dev(slot), in cb710_mmc_set_ios()
580 "powerup retry failed (%d) - expect errors\n", in cb710_mmc_set_ios()
583 reader->last_power_mode = MMC_POWER_ON; in cb710_mmc_set_ios()
586 cb710_mmc_powerdown(slot); in cb710_mmc_set_ios()
587 reader->last_power_mode = MMC_POWER_OFF; in cb710_mmc_set_ios()
596 cb710_mmc_enable_4bit_data(slot, ios->bus_width != MMC_BUS_WIDTH_1); in cb710_mmc_set_ios()
598 cb710_mmc_enable_irq(slot, CB710_MMC_IE_TEST_MASK, 0); in cb710_mmc_set_ios()
601 static int cb710_mmc_get_ro(struct mmc_host *mmc) in cb710_mmc_get_ro() argument
603 struct cb710_slot *slot = cb710_mmc_to_slot(mmc); in cb710_mmc_get_ro() local
605 return cb710_read_port_8(slot, CB710_MMC_STATUS3_PORT) in cb710_mmc_get_ro()
609 static int cb710_mmc_get_cd(struct mmc_host *mmc) in cb710_mmc_get_cd() argument
611 struct cb710_slot *slot = cb710_mmc_to_slot(mmc); in cb710_mmc_get_cd() local
613 return cb710_read_port_8(slot, CB710_MMC_STATUS3_PORT) in cb710_mmc_get_cd()
617 static int cb710_mmc_irq_handler(struct cb710_slot *slot) in cb710_mmc_irq_handler() argument
619 struct mmc_host *mmc = cb710_slot_to_mmc(slot); in cb710_mmc_irq_handler() local
620 struct cb710_mmc_reader *reader = mmc_priv(mmc); in cb710_mmc_irq_handler()
623 status = cb710_read_port_32(slot, CB710_MMC_STATUS_PORT); in cb710_mmc_irq_handler()
624 irqen = cb710_read_port_32(slot, CB710_MMC_IRQ_ENABLE_PORT); in cb710_mmc_irq_handler()
625 config2 = cb710_read_port_32(slot, CB710_MMC_CONFIGB_PORT); in cb710_mmc_irq_handler()
626 config1 = cb710_read_port_32(slot, CB710_MMC_CONFIG_PORT); in cb710_mmc_irq_handler()
628 dev_dbg(cb710_slot_dev(slot), "interrupt; status: %08X, " in cb710_mmc_irq_handler()
634 cb710_write_port_8(slot, CB710_MMC_STATUS1_PORT, in cb710_mmc_irq_handler()
638 mmc_detect_change(mmc, HZ/5); in cb710_mmc_irq_handler()
640 dev_dbg(cb710_slot_dev(slot), "unknown interrupt (test)\n"); in cb710_mmc_irq_handler()
641 spin_lock(&reader->irq_lock); in cb710_mmc_irq_handler()
642 __cb710_mmc_enable_irq(slot, 0, CB710_MMC_IE_TEST_MASK); in cb710_mmc_irq_handler()
643 spin_unlock(&reader->irq_lock); in cb710_mmc_irq_handler()
653 struct mmc_request *mrq = reader->mrq; in cb710_mmc_finish_request_bh_work()
655 reader->mrq = NULL; in cb710_mmc_finish_request_bh_work()
670 struct cb710_slot *slot = cb710_pdev_to_slot(pdev); in cb710_mmc_suspend() local
672 cb710_mmc_enable_irq(slot, 0, ~0); in cb710_mmc_suspend()
678 struct cb710_slot *slot = cb710_pdev_to_slot(pdev); in cb710_mmc_resume() local
680 cb710_mmc_enable_irq(slot, 0, ~0); in cb710_mmc_resume()
688 struct cb710_slot *slot = cb710_pdev_to_slot(pdev); in cb710_mmc_init() local
689 struct cb710_chip *chip = cb710_slot_to_chip(slot); in cb710_mmc_init()
690 struct mmc_host *mmc; in cb710_mmc_init() local
695 mmc = mmc_alloc_host(sizeof(*reader), cb710_slot_dev(slot)); in cb710_mmc_init()
696 if (!mmc) in cb710_mmc_init()
697 return -ENOMEM; in cb710_mmc_init()
699 platform_set_drvdata(pdev, mmc); in cb710_mmc_init()
702 pci_read_config_dword(chip->pdev, 0x48, &val); in cb710_mmc_init()
704 dev_dbg(cb710_slot_dev(slot), "source frequency: %dMHz\n", val); in cb710_mmc_init()
707 mmc->ops = &cb710_mmc_host; in cb710_mmc_init()
708 mmc->f_max = val; in cb710_mmc_init()
709 mmc->f_min = val >> cb710_clock_divider_log2[CB710_MAX_DIVIDER_IDX]; in cb710_mmc_init()
710 mmc->ocr_avail = MMC_VDD_32_33|MMC_VDD_33_34; in cb710_mmc_init()
711 mmc->caps = MMC_CAP_4_BIT_DATA; in cb710_mmc_init()
715 * use of the cmd->busy_timeout. in cb710_mmc_init()
717 mmc->max_busy_timeout = CB710_MMC_REQ_TIMEOUT_MS; in cb710_mmc_init()
719 reader = mmc_priv(mmc); in cb710_mmc_init()
721 INIT_WORK(&reader->finish_req_bh_work, in cb710_mmc_init()
723 spin_lock_init(&reader->irq_lock); in cb710_mmc_init()
726 cb710_mmc_enable_irq(slot, 0, ~0); in cb710_mmc_init()
727 cb710_set_irq_handler(slot, cb710_mmc_irq_handler); in cb710_mmc_init()
729 err = mmc_add_host(mmc); in cb710_mmc_init()
733 dev_dbg(cb710_slot_dev(slot), "mmc_hostname is %s\n", in cb710_mmc_init()
734 mmc_hostname(mmc)); in cb710_mmc_init()
736 cb710_mmc_enable_irq(slot, CB710_MMC_IE_CARD_INSERTION_STATUS, 0); in cb710_mmc_init()
741 dev_dbg(cb710_slot_dev(slot), "mmc_add_host() failed: %d\n", err); in cb710_mmc_init()
743 cb710_set_irq_handler(slot, NULL); in cb710_mmc_init()
744 mmc_free_host(mmc); in cb710_mmc_init()
750 struct cb710_slot *slot = cb710_pdev_to_slot(pdev); in cb710_mmc_exit() local
751 struct mmc_host *mmc = cb710_slot_to_mmc(slot); in cb710_mmc_exit() local
752 struct cb710_mmc_reader *reader = mmc_priv(mmc); in cb710_mmc_exit()
754 cb710_mmc_enable_irq(slot, 0, CB710_MMC_IE_CARD_INSERTION_STATUS); in cb710_mmc_exit()
756 mmc_remove_host(mmc); in cb710_mmc_exit()
759 cb710_mmc_enable_irq(slot, 0, ~0); in cb710_mmc_exit()
760 cb710_set_irq_handler(slot, NULL); in cb710_mmc_exit()
762 /* clear config ports - just in case */ in cb710_mmc_exit()
763 cb710_write_port_32(slot, CB710_MMC_CONFIG_PORT, 0); in cb710_mmc_exit()
764 cb710_write_port_16(slot, CB710_MMC_CONFIGB_PORT, 0); in cb710_mmc_exit()
766 cancel_work_sync(&reader->finish_req_bh_work); in cb710_mmc_exit()
768 mmc_free_host(mmc); in cb710_mmc_exit()
772 .driver.name = "cb710-mmc",
783 MODULE_AUTHOR("Michał Mirosław <mirq-[email protected]>");
784 MODULE_DESCRIPTION("ENE CB710 memory card reader driver - MMC/SD part");
786 MODULE_ALIAS("platform:cb710-mmc");