block_dev.c 14.0 KB
Newer Older
1
/*
2
 * Copyright (c) 2006-2021, RT-Thread Development Team
3
 *
4
 * SPDX-License-Identifier: Apache-2.0
5 6
 *
 * Change Logs:
Y
yiyue.fang 已提交
7 8
 * Date           Author        Notes
 * 2011-07-25     weety     first version
9 10 11 12 13
 */

#include <rtthread.h>
#include <dfs_fs.h>

14
#include <drivers/mmcsd_core.h>
15

16
#define DBG_TAG               "SDIO"
17
#ifdef RT_SDIO_DEBUG
18
#define DBG_LVL               DBG_LOG
19
#else
20
#define DBG_LVL               DBG_INFO
21 22 23
#endif /* RT_SDIO_DEBUG */
#include <rtdbg.h>

24
static rt_list_t blk_devices = RT_LIST_OBJECT_INIT(blk_devices);
25

26 27
#define BLK_MIN(a, b) ((a) < (b) ? (a) : (b))

28 29
struct mmcsd_blk_device
{
Y
yiyue.fang 已提交
30 31 32 33 34
    struct rt_mmcsd_card *card;
    rt_list_t list;
    struct rt_device dev;
    struct dfs_partition part;
    struct rt_device_blk_geometry geometry;
35
    rt_size_t max_req_size;
36 37 38 39 40 41
};

#ifndef RT_MMCSD_MAX_PARTITION
#define RT_MMCSD_MAX_PARTITION 16
#endif

42
rt_int32_t mmcsd_num_wr_blocks(struct rt_mmcsd_card *card)
43
{
Y
yiyue.fang 已提交
44 45
    rt_int32_t err;
    rt_uint32_t blocks;
46

Y
yiyue.fang 已提交
47 48 49 50
    struct rt_mmcsd_req req;
    struct rt_mmcsd_cmd cmd;
    struct rt_mmcsd_data data;
    rt_uint32_t timeout_us;
51

Y
yiyue.fang 已提交
52
    rt_memset(&cmd, 0, sizeof(struct rt_mmcsd_cmd));
53

Y
yiyue.fang 已提交
54 55 56
    cmd.cmd_code = APP_CMD;
    cmd.arg = card->rca << 16;
    cmd.flags = RESP_SPI_R1 | RESP_R1 | CMD_AC;
57

Y
yiyue.fang 已提交
58 59 60 61 62
    err = mmcsd_send_cmd(card->host, &cmd, 0);
    if (err)
        return -RT_ERROR;
    if (!controller_is_spi(card->host) && !(cmd.resp[0] & R1_APP_CMD))
        return -RT_ERROR;
63

Y
yiyue.fang 已提交
64
    rt_memset(&cmd, 0, sizeof(struct rt_mmcsd_cmd));
65

Y
yiyue.fang 已提交
66 67 68
    cmd.cmd_code = SD_APP_SEND_NUM_WR_BLKS;
    cmd.arg = 0;
    cmd.flags = RESP_SPI_R1 | RESP_R1 | CMD_ADTC;
69

Y
yiyue.fang 已提交
70
    rt_memset(&data, 0, sizeof(struct rt_mmcsd_data));
71

Y
yiyue.fang 已提交
72 73
    data.timeout_ns = card->tacc_ns * 100;
    data.timeout_clks = card->tacc_clks * 100;
74

Y
yiyue.fang 已提交
75 76 77
    timeout_us = data.timeout_ns / 1000;
    timeout_us += data.timeout_clks * 1000 /
        (card->host->io_cfg.clock / 1000);
78

79
    if (timeout_us > 100000)
Y
yiyue.fang 已提交
80 81 82 83
    {
        data.timeout_ns = 100000000;
        data.timeout_clks = 0;
    }
84

Y
yiyue.fang 已提交
85 86 87 88
    data.blksize = 4;
    data.blks = 1;
    data.flags = DATA_DIR_READ;
    data.buf = &blocks;
89

Y
yiyue.fang 已提交
90
    rt_memset(&req, 0, sizeof(struct rt_mmcsd_req));
91

Y
yiyue.fang 已提交
92 93
    req.cmd = &cmd;
    req.data = &data;
94

Y
yiyue.fang 已提交
95
    mmcsd_send_request(card->host, &req);
96

Y
yiyue.fang 已提交
97 98
    if (cmd.err || data.err)
        return -RT_ERROR;
99

Y
yiyue.fang 已提交
100
    return blocks;
101 102
}

Y
yiyue.fang 已提交
103 104 105 106 107
static rt_err_t rt_mmcsd_req_blk(struct rt_mmcsd_card *card,
                                 rt_uint32_t           sector,
                                 void                 *buf,
                                 rt_size_t             blks,
                                 rt_uint8_t            dir)
108
{
Y
yiyue.fang 已提交
109 110 111 112 113 114 115 116 117 118 119 120 121
    struct rt_mmcsd_cmd  cmd, stop;
    struct rt_mmcsd_data  data;
    struct rt_mmcsd_req  req;
    struct rt_mmcsd_host *host = card->host;
    rt_uint32_t r_cmd, w_cmd;

    mmcsd_host_lock(host);
    rt_memset(&req, 0, sizeof(struct rt_mmcsd_req));
    rt_memset(&cmd, 0, sizeof(struct rt_mmcsd_cmd));
    rt_memset(&stop, 0, sizeof(struct rt_mmcsd_cmd));
    rt_memset(&data, 0, sizeof(struct rt_mmcsd_data));
    req.cmd = &cmd;
    req.data = &data;
122

Y
yiyue.fang 已提交
123
    cmd.arg = sector;
124
    if (!(card->flags & CARD_FLAG_SDHC))
Y
yiyue.fang 已提交
125 126 127 128 129 130 131 132
    {
        cmd.arg <<= 9;
    }
    cmd.flags = RESP_SPI_R1 | RESP_R1 | CMD_ADTC;

    data.blksize = SECTOR_SIZE;
    data.blks  = blks;

133
    if (blks > 1)
Y
yiyue.fang 已提交
134 135 136 137 138 139 140 141 142 143 144 145 146
    {
        if (!controller_is_spi(card->host) || !dir)
        {
            req.stop = &stop;
            stop.cmd_code = STOP_TRANSMISSION;
            stop.arg = 0;
            stop.flags = RESP_SPI_R1B | RESP_R1B | CMD_AC;
        }
        r_cmd = READ_MULTIPLE_BLOCK;
        w_cmd = WRITE_MULTIPLE_BLOCK;
    }
    else
    {
W
weety 已提交
147
        req.stop = RT_NULL;
Y
yiyue.fang 已提交
148 149 150 151
        r_cmd = READ_SINGLE_BLOCK;
        w_cmd = WRITE_BLOCK;
    }

152
    if (!dir)
Y
yiyue.fang 已提交
153 154 155 156 157 158 159 160 161 162 163 164 165 166
    {
        cmd.cmd_code = r_cmd;
        data.flags |= DATA_DIR_READ;
    }
    else
    {
        cmd.cmd_code = w_cmd;
        data.flags |= DATA_DIR_WRITE;
    }

    mmcsd_set_data_timeout(&data, card);
    data.buf = buf;
    mmcsd_send_request(host, &req);

167
    if (!controller_is_spi(card->host) && dir != 0)
Y
yiyue.fang 已提交
168
    {
169
        do
Y
yiyue.fang 已提交
170 171 172 173 174 175 176
        {
            rt_int32_t err;

            cmd.cmd_code = SEND_STATUS;
            cmd.arg = card->rca << 16;
            cmd.flags = RESP_R1 | CMD_AC;
            err = mmcsd_send_cmd(card->host, &cmd, 5);
177
            if (err)
Y
yiyue.fang 已提交
178
            {
179
                LOG_E("error %d requesting status", err);
Y
yiyue.fang 已提交
180 181 182 183 184 185 186 187 188 189 190 191 192
                break;
            }
            /*
             * Some cards mishandle the status bits,
             * so make sure to check both the busy
             * indication and the card state.
             */
         } while (!(cmd.resp[0] & R1_READY_FOR_DATA) ||
            (R1_CURRENT_STATE(cmd.resp[0]) == 7));
    }

    mmcsd_host_unlock(host);

193
    if (cmd.err || data.err || stop.err)
Y
yiyue.fang 已提交
194
    {
195 196
        LOG_E("mmcsd request blocks error");
        LOG_E("%d,%d,%d, 0x%08x,0x%08x",
Y
yiyue.fang 已提交
197 198 199 200 201 202
                   cmd.err, data.err, stop.err, data.flags, sector);

        return -RT_ERROR;
    }

    return RT_EOK;
203 204 205 206
}

static rt_err_t rt_mmcsd_init(rt_device_t dev)
{
Y
yiyue.fang 已提交
207
    return RT_EOK;
208 209 210 211
}

static rt_err_t rt_mmcsd_open(rt_device_t dev, rt_uint16_t oflag)
{
Y
yiyue.fang 已提交
212
    return RT_EOK;
213 214 215 216
}

static rt_err_t rt_mmcsd_close(rt_device_t dev)
{
Y
yiyue.fang 已提交
217
    return RT_EOK;
218 219
}

B
bernard 已提交
220
static rt_err_t rt_mmcsd_control(rt_device_t dev, int cmd, void *args)
221
{
Y
yiyue.fang 已提交
222 223 224 225 226 227 228 229 230 231
    struct mmcsd_blk_device *blk_dev = (struct mmcsd_blk_device *)dev->user_data;
    switch (cmd)
    {
    case RT_DEVICE_CTRL_BLK_GETGEOME:
        rt_memcpy(args, &blk_dev->geometry, sizeof(struct rt_device_blk_geometry));
        break;
    default:
        break;
    }
    return RT_EOK;
232 233
}

Y
yiyue.fang 已提交
234 235 236 237
static rt_size_t rt_mmcsd_read(rt_device_t dev,
                               rt_off_t    pos,
                               void       *buffer,
                               rt_size_t   size)
238
{
239
    rt_err_t err = 0;
240 241 242 243
    rt_size_t offset = 0;
    rt_size_t req_size = 0;
    rt_size_t remain_size = size;
    void *rd_ptr = (void *)buffer;
Y
yiyue.fang 已提交
244 245 246 247 248
    struct mmcsd_blk_device *blk_dev = (struct mmcsd_blk_device *)dev->user_data;
    struct dfs_partition *part = &blk_dev->part;

    if (dev == RT_NULL)
    {
B
bernard 已提交
249
        rt_set_errno(-EINVAL);
Y
yiyue.fang 已提交
250 251 252 253
        return 0;
    }

    rt_sem_take(part->lock, RT_WAITING_FOREVER);
254 255
    while (remain_size)
    {
256
        req_size = (remain_size > blk_dev->max_req_size) ? blk_dev->max_req_size : remain_size;
257 258 259 260
        err = rt_mmcsd_req_blk(blk_dev->card, part->offset + pos + offset, rd_ptr, req_size, 0);
        if (err)
            break;
        offset += req_size;
W
weety 已提交
261
        rd_ptr = (void *)((rt_uint8_t *)rd_ptr + (req_size << 9));
262 263
        remain_size -= req_size;
    }
Y
yiyue.fang 已提交
264 265 266
    rt_sem_release(part->lock);

    /* the length of reading must align to SECTOR SIZE */
267
    if (err)
Y
yiyue.fang 已提交
268
    {
B
bernard 已提交
269
        rt_set_errno(-EIO);
Y
yiyue.fang 已提交
270 271
        return 0;
    }
272
    return size - remain_size;
273 274
}

Y
yiyue.fang 已提交
275 276 277 278
static rt_size_t rt_mmcsd_write(rt_device_t dev,
                                rt_off_t    pos,
                                const void *buffer,
                                rt_size_t   size)
279
{
280
    rt_err_t err = 0;
281 282 283 284
    rt_size_t offset = 0;
    rt_size_t req_size = 0;
    rt_size_t remain_size = size;
    void *wr_ptr = (void *)buffer;
Y
yiyue.fang 已提交
285 286 287 288 289
    struct mmcsd_blk_device *blk_dev = (struct mmcsd_blk_device *)dev->user_data;
    struct dfs_partition *part = &blk_dev->part;

    if (dev == RT_NULL)
    {
B
bernard 已提交
290
        rt_set_errno(-EINVAL);
Y
yiyue.fang 已提交
291 292 293 294
        return 0;
    }

    rt_sem_take(part->lock, RT_WAITING_FOREVER);
295 296
    while (remain_size)
    {
297
        req_size = (remain_size > blk_dev->max_req_size) ? blk_dev->max_req_size : remain_size;
298 299 300 301
        err = rt_mmcsd_req_blk(blk_dev->card, part->offset + pos + offset, wr_ptr, req_size, 1);
        if (err)
            break;
        offset += req_size;
W
weety 已提交
302
        wr_ptr = (void *)((rt_uint8_t *)wr_ptr + (req_size << 9));
303 304
        remain_size -= req_size;
    }
Y
yiyue.fang 已提交
305 306 307
    rt_sem_release(part->lock);

    /* the length of reading must align to SECTOR SIZE */
308
    if (err)
Y
yiyue.fang 已提交
309
    {
B
bernard 已提交
310
        rt_set_errno(-EIO);
Y
yiyue.fang 已提交
311 312 313

        return 0;
    }
314
    return size - remain_size;
315 316 317 318
}

static rt_int32_t mmcsd_set_blksize(struct rt_mmcsd_card *card)
{
Y
yiyue.fang 已提交
319 320 321 322 323 324 325 326 327 328 329 330 331 332
    struct rt_mmcsd_cmd cmd;
    int err;

    /* Block-addressed cards ignore MMC_SET_BLOCKLEN. */
    if (card->flags & CARD_FLAG_SDHC)
        return 0;

    mmcsd_host_lock(card->host);
    cmd.cmd_code = SET_BLOCKLEN;
    cmd.arg = 512;
    cmd.flags = RESP_SPI_R1 | RESP_R1 | CMD_AC;
    err = mmcsd_send_cmd(card->host, &cmd, 5);
    mmcsd_host_unlock(card->host);

333
    if (err)
Y
yiyue.fang 已提交
334
    {
335
        LOG_E("MMCSD: unable to set block size to %d: %d", cmd.arg, err);
Y
yiyue.fang 已提交
336 337 338 339 340

        return -RT_ERROR;
    }

    return 0;
341 342
}

B
Bernard Xiong 已提交
343
#ifdef RT_USING_DEVICE_OPS
344
const static struct rt_device_ops mmcsd_blk_ops =
B
Bernard Xiong 已提交
345 346 347 348 349 350 351 352 353 354
{
    rt_mmcsd_init,
    rt_mmcsd_open,
    rt_mmcsd_close,
    rt_mmcsd_read,
    rt_mmcsd_write,
    rt_mmcsd_control
};
#endif

355 356
rt_int32_t rt_mmcsd_blk_probe(struct rt_mmcsd_card *card)
{
Y
yiyue.fang 已提交
357 358 359 360 361 362 363 364
    rt_int32_t err = 0;
    rt_uint8_t i, status;
    rt_uint8_t *sector;
    char dname[4];
    char sname[8];
    struct mmcsd_blk_device *blk_dev = RT_NULL;

    err = mmcsd_set_blksize(card);
365
    if(err)
Y
yiyue.fang 已提交
366 367 368 369
    {
        return err;
    }

370
    LOG_D("probe mmcsd block device!");
371

Y
yiyue.fang 已提交
372 373 374 375
    /* get the first sector to read partition table */
    sector = (rt_uint8_t *)rt_malloc(SECTOR_SIZE);
    if (sector == RT_NULL)
    {
376
        LOG_E("allocate partition sector buffer failed!");
Y
yiyue.fang 已提交
377 378 379 380 381 382 383 384 385

        return -RT_ENOMEM;
    }

    status = rt_mmcsd_req_blk(card, 0, sector, 1, 0);
    if (status == RT_EOK)
    {
        for (i = 0; i < RT_MMCSD_MAX_PARTITION; i++)
        {
386
            blk_dev = rt_calloc(1, sizeof(struct mmcsd_blk_device));
387
            if (!blk_dev)
Y
yiyue.fang 已提交
388
            {
389
                LOG_E("mmcsd:malloc memory failed!");
Y
yiyue.fang 已提交
390 391
                break;
            }
392

393 394 395
            blk_dev->max_req_size = BLK_MIN((card->host->max_dma_segs *
                                             card->host->max_seg_size) >> 9,
                                            (card->host->max_blk_count *
396 397
                                             card->host->max_blk_size) >> 9);

Y
yiyue.fang 已提交
398 399 400 401 402 403 404
            /* get the first partition */
            status = dfs_filesystem_get_partition(&blk_dev->part, sector, i);
            if (status == RT_EOK)
            {
                rt_snprintf(dname, 4, "sd%d",  i);
                rt_snprintf(sname, 8, "sem_sd%d",  i);
                blk_dev->part.lock = rt_sem_create(sname, 1, RT_IPC_FLAG_FIFO);
405

Y
yiyue.fang 已提交
406
                /* register mmcsd device */
B
Bernard Xiong 已提交
407 408 409 410
                blk_dev->dev.type = RT_Device_Class_Block;
#ifdef RT_USING_DEVICE_OPS
                blk_dev->dev.ops  = &mmcsd_blk_ops;
#else
Y
yiyue.fang 已提交
411 412 413 414 415 416
                blk_dev->dev.init = rt_mmcsd_init;
                blk_dev->dev.open = rt_mmcsd_open;
                blk_dev->dev.close = rt_mmcsd_close;
                blk_dev->dev.read = rt_mmcsd_read;
                blk_dev->dev.write = rt_mmcsd_write;
                blk_dev->dev.control = rt_mmcsd_control;
B
Bernard Xiong 已提交
417
#endif
Y
yiyue.fang 已提交
418 419 420
                blk_dev->dev.user_data = blk_dev;

                blk_dev->card = card;
421

Y
yiyue.fang 已提交
422 423 424
                blk_dev->geometry.bytes_per_sector = 1<<9;
                blk_dev->geometry.block_size = card->card_blksize;
                blk_dev->geometry.sector_count = blk_dev->part.size;
425

Y
yiyue.fang 已提交
426 427 428 429 430 431 432 433 434 435 436 437
                rt_device_register(&blk_dev->dev, dname,
                    RT_DEVICE_FLAG_RDWR | RT_DEVICE_FLAG_REMOVABLE | RT_DEVICE_FLAG_STANDALONE);
                rt_list_insert_after(&blk_devices, &blk_dev->list);
            }
            else
            {
                if (i == 0)
                {
                    /* there is no partition table */
                    blk_dev->part.offset = 0;
                    blk_dev->part.size   = 0;
                    blk_dev->part.lock = rt_sem_create("sem_sd0", 1, RT_IPC_FLAG_FIFO);
438

Y
yiyue.fang 已提交
439 440
                    /* register mmcsd device */
                    blk_dev->dev.type  = RT_Device_Class_Block;
B
Bernard Xiong 已提交
441 442 443
#ifdef RT_USING_DEVICE_OPS
                    blk_dev->dev.ops  = &mmcsd_blk_ops;
#else
Y
yiyue.fang 已提交
444 445 446 447 448 449
                    blk_dev->dev.init = rt_mmcsd_init;
                    blk_dev->dev.open = rt_mmcsd_open;
                    blk_dev->dev.close = rt_mmcsd_close;
                    blk_dev->dev.read = rt_mmcsd_read;
                    blk_dev->dev.write = rt_mmcsd_write;
                    blk_dev->dev.control = rt_mmcsd_control;
B
Bernard Xiong 已提交
450
#endif
Y
yiyue.fang 已提交
451 452 453 454 455 456
                    blk_dev->dev.user_data = blk_dev;

                    blk_dev->card = card;

                    blk_dev->geometry.bytes_per_sector = 1<<9;
                    blk_dev->geometry.block_size = card->card_blksize;
457
                    blk_dev->geometry.sector_count =
W
weety 已提交
458
                        card->card_capacity * (1024 / 512);
459

Y
yiyue.fang 已提交
460 461 462 463 464 465 466 467 468 469 470
                    rt_device_register(&blk_dev->dev, "sd0",
                        RT_DEVICE_FLAG_RDWR | RT_DEVICE_FLAG_REMOVABLE | RT_DEVICE_FLAG_STANDALONE);
                    rt_list_insert_after(&blk_devices, &blk_dev->list);
                }
                else
                {
                    rt_free(blk_dev);
                    blk_dev = RT_NULL;
                    break;
                }
            }
471 472

#ifdef RT_USING_DFS_MNTTABLE
H
hichard 已提交
473
            if (blk_dev)
474
            {
475 476 477
                LOG_I("try to mount file system!");
                /* try to mount file system on this block device */
                dfs_mount_device(&(blk_dev->dev));
478 479
            }
#endif
Y
yiyue.fang 已提交
480 481 482 483
        }
    }
    else
    {
484
        LOG_E("read mmcsd first sector failed");
Y
yiyue.fang 已提交
485 486
        err = -RT_ERROR;
    }
487

Y
yiyue.fang 已提交
488 489
    /* release sector buffer */
    rt_free(sector);
490

Y
yiyue.fang 已提交
491
    return err;
492 493 494 495
}

void rt_mmcsd_blk_remove(struct rt_mmcsd_card *card)
{
496
    rt_list_t *l, *n;
Y
yiyue.fang 已提交
497
    struct mmcsd_blk_device *blk_dev;
498

499
    for (l = (&blk_devices)->next, n = l->next; l != &blk_devices; l = n, n = n->next)
Y
yiyue.fang 已提交
500 501
    {
        blk_dev = (struct mmcsd_blk_device *)rt_list_entry(l, struct mmcsd_blk_device, list);
502
        if (blk_dev->card == card)
Y
yiyue.fang 已提交
503
        {
504 505 506 507
            /* unmount file system */
            const char * mounted_path = dfs_filesystem_get_mounted_path(&(blk_dev->dev));
            if (mounted_path)
            {
H
hichard 已提交
508 509
                  dfs_unmount(mounted_path);
                  LOG_D("unmount file system %s for device %s.\r\n", mounted_path, blk_dev->dev.parent.name);
510
            }
H
hichard 已提交
511
            rt_sem_delete(blk_dev->part.lock);
Y
yiyue.fang 已提交
512 513 514 515 516
            rt_device_unregister(&blk_dev->dev);
            rt_list_remove(&blk_dev->list);
            rt_free(blk_dev);
        }
    }
517
}