mirror of
https://github.com/Stichting-MINIX-Research-Foundation/netbsd.git
synced 2025-09-10 15:46:33 -04:00
455 lines
12 KiB
C
455 lines
12 KiB
C
/*-
|
|
* Copyright (c) 2012 The NetBSD Foundation, Inc.
|
|
* All rights reserved.
|
|
*
|
|
* This code is derived from software contributed to The NetBSD Foundation
|
|
* by Paul Fleischer <paul@xpg.dk>
|
|
*
|
|
* Redistribution and use in source and binary forms, with or without
|
|
* modification, are permitted provided that the following conditions
|
|
* are met:
|
|
* 1. Redistributions of source code must retain the above copyright
|
|
* notice, this list of conditions and the following disclaimer.
|
|
* 2. Redistributions in binary form must reproduce the above copyright
|
|
* notice, this list of conditions and the following disclaimer in the
|
|
* documentation and/or other materials provided with the distribution.
|
|
*
|
|
* THIS SOFTWARE IS PROVIDED BY THE NETBSD FOUNDATION, INC. AND CONTRIBUTORS
|
|
* ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED
|
|
* TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
|
|
* PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE FOUNDATION OR CONTRIBUTORS
|
|
* BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
|
|
* CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
|
|
* SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
|
|
* INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
|
|
* CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
|
|
* ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
|
|
* POSSIBILITY OF SUCH DAMAGE.
|
|
*/
|
|
#include "s3csdi.h"
|
|
|
|
#include <arm/s3c2xx0/s3c2440reg.h>
|
|
|
|
#include <lib/libsa/stand.h>
|
|
|
|
#include <machine/int_mwgwtypes.h>
|
|
#include <machine/limits.h>
|
|
|
|
#include <dev/sdmmc/sdmmcreg.h>
|
|
|
|
#define SDI_REG(reg) (*(volatile uint32_t*)(S3C2440_SDI_BASE+reg))
|
|
|
|
//#define SSSDI_DEBUG
|
|
#ifdef SSSDI_DEBUG
|
|
#define DPRINTF(s) do {printf s; } while (/*CONSTCOND*/0)
|
|
#else
|
|
#define DPRINTF(s) do {} while (/*CONSTCOND*/0)
|
|
#endif
|
|
|
|
struct s3csdi_softc {
|
|
int width;
|
|
};
|
|
|
|
extern int pclk;
|
|
|
|
static void sssdi_perform_pio_read(struct sdmmc_command *cmd);
|
|
//static void sssdi_perform_pio_write(struct sdmmc_command *cmd);
|
|
|
|
static struct s3csdi_softc s3csdi_softc;
|
|
|
|
int
|
|
s3csd_match(unsigned int tag)
|
|
{
|
|
printf("Found S3C2440 SD/MMC\n");
|
|
return 1;
|
|
}
|
|
|
|
void*
|
|
s3csd_init(unsigned int tag, uint32_t *caps)
|
|
{
|
|
uint32_t data;
|
|
|
|
*caps = SMC_CAPS_4BIT_MODE;
|
|
|
|
DPRINTF(("CLKCON: 0x%X\n", *(volatile uint32_t*)(S3C2440_CLKMAN_BASE + CLKMAN_CLKCON)));
|
|
|
|
DPRINTF(("SDI_INT_MASK: 0x%X\n", SDI_REG(SDI_INT_MASK)));
|
|
|
|
SDI_REG(SDI_INT_MASK) = 0x0;
|
|
SDI_REG(SDI_DTIMER) = 0x007FFFFF;
|
|
|
|
SDI_REG(SDI_CON) &= ~SDICON_ENCLK;
|
|
|
|
SDI_REG(SDI_CON) = SDICON_SD_RESET | SDICON_CTYP_SD;
|
|
|
|
/* Set GPG8 to input such that we can check if there is a card present
|
|
*/
|
|
data = *(volatile uint32_t*)(S3C2440_GPIO_BASE+GPIO_PGCON);
|
|
data = GPIO_SET_FUNC(data, 8, 0x00);
|
|
*(volatile uint32_t*)(S3C2440_GPIO_BASE+GPIO_PGCON) = data;
|
|
|
|
/* Check if a card is present */
|
|
data = *(volatile uint32_t*)(S3C2440_GPIO_BASE+GPIO_PGDAT);
|
|
if ( (data & (1<<8)) == 1) {
|
|
printf("No card detected\n");
|
|
/* Pin 8 is low when no card is inserted */
|
|
return 0;
|
|
}
|
|
printf("Card detected\n");
|
|
|
|
s3csdi_softc.width = 1;
|
|
|
|
/* We have no private data to return, but 0 signals error */
|
|
return (void*)0x01;
|
|
}
|
|
|
|
int
|
|
s3csd_bus_clock(void *priv, int freq)
|
|
{
|
|
int div;
|
|
int clock_set = 0;
|
|
int control;
|
|
int clk = pclk/1000; /*Peripheral bus clock in KHz*/
|
|
|
|
/* Round peripheral bus clock down to nearest MHz */
|
|
clk = (clk / 1000) * 1000;
|
|
|
|
control = SDI_REG(SDI_CON);
|
|
SDI_REG(SDI_CON) = control & ~SDICON_ENCLK;
|
|
|
|
|
|
/* If the frequency is zero just keep the clock disabled */
|
|
if (freq == 0)
|
|
return 0;
|
|
|
|
for (div = 1; div <= 256; div++) {
|
|
if ( clk / div <= freq) {
|
|
DPRINTF(("Using divisor %d: %d/%d = %d\n", div, clk,
|
|
div, clk/div));
|
|
clock_set = 1;
|
|
SDI_REG(SDI_PRE) = div-1;
|
|
break;
|
|
}
|
|
}
|
|
|
|
if (clock_set) {
|
|
SDI_REG(SDI_CON) = control | SDICON_ENCLK;
|
|
if (div-1 != SDI_REG(SDI_PRE)) {
|
|
return 1;
|
|
}
|
|
|
|
sdmmc_delay(74000/freq);
|
|
/* Wait for 74 SDCLK */
|
|
/* 1/freq is the length of a clock cycle,
|
|
so we have to wait 1/freq * 74 .
|
|
74000 / freq should express the delay in us.
|
|
*/
|
|
return 0;
|
|
} else {
|
|
return 1;
|
|
}
|
|
}
|
|
|
|
#define SSSDI_TRANSFER_NONE 0
|
|
#define SSSDI_TRANSFER_READ 1
|
|
#define SSSDI_TRANSFER_WRITE 2
|
|
|
|
void
|
|
s3csd_exec_cmd(void *priv, struct sdmmc_command *cmd)
|
|
{
|
|
uint32_t cmd_control;
|
|
int status = 0;
|
|
uint32_t data_status;
|
|
int transfer = SSSDI_TRANSFER_NONE;
|
|
|
|
DPRINTF(("s3csd_exec_cmd\n"));
|
|
|
|
SDI_REG(SDI_DAT_FSTA) = 0xFFFFFFFF;
|
|
SDI_REG(SDI_DAT_STA) = 0xFFFFFFFF;
|
|
SDI_REG(SDI_CMD_STA) = 0xFFFFFFFF;
|
|
|
|
SDI_REG(SDI_CMD_ARG) = cmd->c_arg;
|
|
|
|
cmd_control = (cmd->c_opcode & SDICMDCON_CMD_MASK) |
|
|
SDICMDCON_HOST_CMD | SDICMDCON_CMST;
|
|
if (cmd->c_flags & SCF_RSP_PRESENT)
|
|
cmd_control |= SDICMDCON_WAIT_RSP;
|
|
if (cmd->c_flags & SCF_RSP_136)
|
|
cmd_control |= SDICMDCON_LONG_RSP;
|
|
|
|
if (cmd->c_datalen > 0 && cmd->c_data != NULL) {
|
|
/* TODO: Ensure that the above condition matches the semantics
|
|
of SDICMDCON_WITH_DATA*/
|
|
DPRINTF(("DATA, datalen: %d, blk_size: %d, offset: %d\n", cmd->c_datalen,
|
|
cmd->c_blklen, cmd->c_arg));
|
|
cmd_control |= SDICMDCON_WITH_DATA;
|
|
}
|
|
|
|
if (cmd->c_opcode == MMC_STOP_TRANSMISSION) {
|
|
cmd_control |= SDICMDCON_ABORT_CMD;
|
|
}
|
|
|
|
SDI_REG(SDI_DTIMER) = 0x007FFFFF;
|
|
SDI_REG(SDI_BSIZE) = cmd->c_blklen;
|
|
|
|
if ( (cmd->c_flags & SCF_CMD_READ) &&
|
|
(cmd_control & SDICMDCON_WITH_DATA)) {
|
|
uint32_t data_control;
|
|
DPRINTF(("Reading %d bytes\n", cmd->c_datalen));
|
|
transfer = SSSDI_TRANSFER_READ;
|
|
|
|
data_control = SDIDATCON_DATMODE_RECEIVE | SDIDATCON_RACMD |
|
|
SDIDATCON_DTST | SDIDATCON_BLKMODE |
|
|
((cmd->c_datalen / cmd->c_blklen) & SDIDATCON_BLKNUM_MASK) |
|
|
SDIDATCON_DATA_WORD;
|
|
|
|
|
|
if (s3csdi_softc.width == 4) {
|
|
data_control |= SDIDATCON_WIDEBUS;
|
|
}
|
|
|
|
SDI_REG(SDI_DAT_CON) = data_control;
|
|
} else if (cmd_control & SDICMDCON_WITH_DATA) {
|
|
/* Write data */
|
|
|
|
uint32_t data_control;
|
|
DPRINTF(("Writing %d bytes\n", cmd->c_datalen));
|
|
DPRINTF(("Requesting %d blocks\n",
|
|
cmd->c_datalen / cmd->c_blklen));
|
|
transfer = SSSDI_TRANSFER_WRITE;
|
|
data_control = SDIDATCON_DATMODE_TRANSMIT | SDIDATCON_BLKMODE |
|
|
SDIDATCON_TARSP | SDIDATCON_DTST |
|
|
((cmd->c_datalen / cmd->c_blklen) & SDIDATCON_BLKNUM_MASK) |
|
|
SDIDATCON_DATA_WORD;
|
|
|
|
/* if (sc->width == 4) {
|
|
data_control |= SDIDATCON_WIDEBUS;
|
|
}*/
|
|
|
|
SDI_REG(SDI_DAT_CON) = data_control;
|
|
}
|
|
|
|
DPRINTF(("SID_CMD_CON: 0x%X\n", cmd_control));
|
|
/* Send command to SDI */
|
|
SDI_REG(SDI_CMD_CON) = cmd_control;
|
|
DPRINTF(("Status before cmd sent: 0x%X\n", SDI_REG(SDI_CMD_STA)));
|
|
DPRINTF(("Waiting for command being sent\n"));
|
|
while( !(SDI_REG(SDI_CMD_STA) & SDICMDSTA_CMD_SENT));
|
|
DPRINTF(("Command has been sent\n"));
|
|
|
|
//SDI_REG(SDI_CMD_STA) |= SDICMDSTA_CMD_SENT;
|
|
|
|
if (!(cmd_control & SDICMDCON_WAIT_RSP)) {
|
|
SDI_REG(SDI_CMD_STA) |= SDICMDSTA_CMD_SENT;
|
|
cmd->c_flags |= SCF_ITSDONE;
|
|
goto out;
|
|
}
|
|
|
|
DPRINTF(("waiting for response\n"));
|
|
while(1) {
|
|
status = SDI_REG(SDI_CMD_STA);
|
|
if (status & SDICMDSTA_RSP_FIN) {
|
|
break;
|
|
}
|
|
if (status & SDICMDSTA_CMD_TIMEOUT) {
|
|
break;
|
|
}
|
|
}
|
|
|
|
DPRINTF(("Status: 0x%X\n", status));
|
|
if (status & SDICMDSTA_CMD_TIMEOUT) {
|
|
cmd->c_error = ETIMEDOUT;
|
|
DPRINTF(("Timeout waiting for response\n"));
|
|
goto out;
|
|
}
|
|
DPRINTF(("Got Response\n"));
|
|
|
|
if (cmd->c_flags & SCF_RSP_136 ) {
|
|
uint32_t w[4];
|
|
|
|
/* We store the response least significant word first */
|
|
w[0] = SDI_REG(SDI_RSP3);
|
|
w[1] = SDI_REG(SDI_RSP2);
|
|
w[2] = SDI_REG(SDI_RSP1);
|
|
w[3] = SDI_REG(SDI_RSP0);
|
|
|
|
/* The sdmmc subsystem expects that the response is delivered
|
|
without the lower 8 bits (CRC + '1' bit) */
|
|
cmd->c_resp[0] = (w[0] >> 8) | ((w[1] & 0xFF) << 24);
|
|
cmd->c_resp[1] = (w[1] >> 8) | ((w[2] & 0XFF) << 24);
|
|
cmd->c_resp[2] = (w[2] >> 8) | ((w[3] & 0XFF) << 24);
|
|
cmd->c_resp[3] = (w[3] >> 8);
|
|
|
|
} else {
|
|
cmd->c_resp[0] = SDI_REG(SDI_RSP0);
|
|
cmd->c_resp[1] = SDI_REG(SDI_RSP1);
|
|
}
|
|
|
|
DPRINTF(("Response: %X %X %X %X\n",
|
|
cmd->c_resp[0],
|
|
cmd->c_resp[1],
|
|
cmd->c_resp[2],
|
|
cmd->c_resp[3]));
|
|
|
|
status = SDI_REG(SDI_DAT_CNT);
|
|
|
|
DPRINTF(("Remaining bytes of current block: %d\n",
|
|
SDIDATCNT_BLK_CNT(status)));
|
|
DPRINTF(("Remaining Block Number : %d\n",
|
|
SDIDATCNT_BLK_NUM_CNT(status)));
|
|
|
|
data_status = SDI_REG(SDI_DAT_STA);
|
|
|
|
DPRINTF(("SDI Data Status Register Before xfer: 0x%X\n", data_status));
|
|
|
|
if (data_status & SDIDATSTA_DATA_TIMEOUT) {
|
|
cmd->c_error = ETIMEDOUT;
|
|
DPRINTF(("Timeout waiting for data\n"));
|
|
goto out;
|
|
}
|
|
|
|
|
|
if (transfer == SSSDI_TRANSFER_READ) {
|
|
DPRINTF(("Waiting for transfer to complete\n"));
|
|
|
|
sssdi_perform_pio_read(cmd);
|
|
} else if (transfer == SSSDI_TRANSFER_WRITE) {
|
|
|
|
/* DPRINTF(("PIO WRITE\n"));
|
|
sssdi_perform_pio_write(sc, cmd);
|
|
|
|
if (cmd->c_error == ETIMEDOUT)
|
|
goto out;*/
|
|
}
|
|
|
|
|
|
/* Response has been received, and any data transfer needed has been
|
|
performed */
|
|
cmd->c_flags |= SCF_ITSDONE;
|
|
|
|
out:
|
|
|
|
data_status = SDI_REG(SDI_DAT_STA);
|
|
DPRINTF(("SDI Data Status Register after execute: 0x%X\n", data_status));
|
|
|
|
/* Clear status register. Their are cleared on the
|
|
next sssdi_exec_command */
|
|
SDI_REG(SDI_CMD_STA) = 0xFFFFFFFF;
|
|
SDI_REG(SDI_DAT_CON) = 0x0;
|
|
}
|
|
|
|
void
|
|
sssdi_perform_pio_read(struct sdmmc_command *cmd)
|
|
{
|
|
uint32_t status;
|
|
uint32_t fifo_status;
|
|
int count;
|
|
uint32_t written;
|
|
uint8_t *dest = (uint8_t*)cmd->c_data;
|
|
int i;
|
|
|
|
written = 0;
|
|
|
|
while (written < cmd->c_datalen ) {
|
|
/* Wait until the FIFO is full or has the final data.
|
|
In the latter case it might not get filled. */
|
|
//status = sssdi_wait_intr(sc, SDI_FIFO_RX_FULL | SDI_FIFO_RX_LAST, 1000);
|
|
//printf("Waiting for FIFO (got %d / %d)\n", written, cmd->c_datalen);
|
|
do {
|
|
status = SDI_REG(SDI_DAT_FSTA);
|
|
} while( !(status & SDIDATFSTA_RF_FULL) && !(status & SDIDATFSTA_RF_LAST));
|
|
//printf("Done\n");
|
|
|
|
fifo_status = SDI_REG(SDI_DAT_FSTA);
|
|
count = SDIDATFSTA_FFCNT(fifo_status);
|
|
|
|
//printf("Writing %d bytes to %p\n", count, dest);
|
|
for(i=0; i<count; i+=4) {
|
|
uint32_t buf;
|
|
|
|
buf = SDI_REG(SDI_DAT_LI_W);
|
|
*dest = (buf & 0xFF); dest++;
|
|
*dest = (buf >> 8) & 0xFF; dest++;
|
|
*dest = (buf >> 16) & 0xFF; dest++;
|
|
*dest = (buf >> 24) & 0xFF; dest++;
|
|
written += 4;
|
|
}
|
|
}
|
|
}
|
|
|
|
#if 0
|
|
void
|
|
sssdi_perform_pio_write(struct sdmmc_command *cmd)
|
|
{
|
|
uint32_t status;
|
|
uint32_t fifo_status;
|
|
int count;
|
|
uint32_t written;
|
|
uint32_t *dest = (uint32_t*)cmd->c_data;
|
|
|
|
written = 0;
|
|
|
|
while (written < cmd->c_datalen ) {
|
|
/* Wait until the FIFO is full or has the final data.
|
|
In the latter case it might not get filled. */
|
|
DPRINTF(("Waiting for FIFO to become empty\n"));
|
|
status = sssdi_wait_intr(sc, SDI_FIFO_TX_EMPTY, 1000);
|
|
|
|
fifo_status = bus_space_read_4(sc->iot, sc->ioh, SDI_DAT_FSTA);
|
|
DPRINTF(("PIO Write FIFO Status: 0x%X\n", fifo_status));
|
|
count = 64-SDIDATFSTA_FFCNT(fifo_status);
|
|
|
|
status = bus_space_read_4(sc->iot, sc->ioh, SDI_DAT_CNT);
|
|
DPRINTF(("Remaining bytes of current block: %d\n",
|
|
SDIDATCNT_BLK_CNT(status)));
|
|
DPRINTF(("Remaining Block Number : %d\n",
|
|
SDIDATCNT_BLK_NUM_CNT(status)));
|
|
|
|
|
|
status = bus_space_read_4(sc->iot,sc->ioh, SDI_DAT_STA);
|
|
DPRINTF(("PIO Write Data Status: 0x%X\n", status));
|
|
|
|
if (status & SDIDATSTA_DATA_TIMEOUT) {
|
|
cmd->c_error = ETIMEDOUT;
|
|
/* Acknowledge the timeout*/
|
|
bus_space_write_4(sc->iot, sc->ioh, SDI_DAT_STA,
|
|
SDIDATSTA_DATA_TIMEOUT);
|
|
printf("%s: Data timeout\n", device_xname(sc->dev));
|
|
break;
|
|
}
|
|
|
|
DPRINTF(("Filling FIFO with %d bytes\n", count));
|
|
for(int i=0; i<count; i+=4) {
|
|
bus_space_write_4(sc->iot, sc->ioh, SDI_DAT_LI_W, *dest);
|
|
written += 4;
|
|
dest++;
|
|
}
|
|
}
|
|
}
|
|
#endif
|
|
|
|
int
|
|
s3csd_host_ocr(void *priv)
|
|
{
|
|
return MMC_OCR_3_2V_3_3V | MMC_OCR_3_3V_3_4V;
|
|
}
|
|
|
|
int
|
|
s3csd_bus_power(void *priv, int ocr)
|
|
{
|
|
return 0;
|
|
}
|
|
|
|
int
|
|
s3csd_bus_width(void *priv, int width)
|
|
{
|
|
s3csdi_softc.width = width;
|
|
return 0;
|
|
}
|
|
|
|
int
|
|
s3csd_get_max_bus_clock(void *priv)
|
|
{
|
|
return pclk / 1;
|
|
}
|