blob: 07a21ea9ad1c88bf60fc493470d86abc02ba5da6 [file] [log] [blame]
/*
* Copyright (c) 2011 The Chromium OS Authors. All rights reserved.
* Use of this source code is governed by a BSD-style license that can be
* found in the LICENSE file.
*
* Alternatively, this software may be distributed under the terms of the
* GNU General Public License ("GPL") version 2 as published by the Free
* Software Foundation.
*/
#include <common.h>
#include <mmc.h>
#include <cros/common.h>
#include <cros/boot_device.h>
static int boot_device_mmc_start(uint32_t disk_flags)
{
/* We expect to have at least one MMC device */
return 1;
}
static int boot_device_mmc_scan(block_dev_desc_t **desc, int max_devs,
uint32_t disk_flags)
{
int index, found;
for (index = found = 0; index < max_devs; index++) {
struct mmc *mmc;
uint32_t flags;
mmc = find_mmc_device(index);
if (!mmc)
break;
/*
* If device flags do not match, we do not even care whether
* the device is presented or not, and simply skip it.
*/
if (!boot_device_matches(&mmc->block_dev, disk_flags, &flags))
continue;
/* Rescan removable device only */
if (mmc->block_dev.removable)
mmc->has_init = 0;
/* Skip device that cannot be initialized */
if (!mmc->has_init && mmc_init(mmc)) {
VBDEBUG("mmc_init fail\n");
continue;
}
/* Skip not-presented device */
if (!mmc->block_dev.lba)
continue;
/*
* find_mmc_device() returns a pointer from a global
* linked list. So &mmc->block_dev is not a dangling
* pointer after this function is returned.
*/
desc[found++] = &mmc->block_dev;
}
return found;
}
static struct boot_interface mmc_interface = {
.name = "mmc",
.type = IF_TYPE_MMC,
.start = boot_device_mmc_start,
.scan = boot_device_mmc_scan,
};
int boot_device_mmc_probe(void)
{
return boot_device_register_interface(&mmc_interface);
}