blob: 9ae02730685aa0e4edf7db664b6f9589514c0786 [file] [log] [blame]
Tom Rini10e47792018-05-06 17:58:06 -04001// SPDX-License-Identifier: GPL-2.0+
Dan Murphy0c3ab852014-02-03 06:59:01 -06002/*
3 * (C) Copyright 2013
4 * Texas Instruments, <www.ti.com>
5 *
6 * Dan Murphy <dmurphy@ti.com>
7 *
Dan Murphy0c3ab852014-02-03 06:59:01 -06008 * Derived work from spl_usb.c
9 */
10
11#include <common.h>
12#include <spl.h>
13#include <asm/u-boot.h>
14#include <sata.h>
Tom Rinieac5b832015-01-05 21:14:04 -050015#include <scsi.h>
Nikita Kiryanov33eefe42015-11-08 17:11:49 +020016#include <errno.h>
Dan Murphy0c3ab852014-02-03 06:59:01 -060017#include <fat.h>
Dan Murphy0c3ab852014-02-03 06:59:01 -060018#include <image.h>
19
Baruch Siachb5224642019-05-16 13:03:53 +030020#ifndef CONFIG_SYS_SATA_FAT_BOOT_PARTITION
21#define CONFIG_SYS_SATA_FAT_BOOT_PARTITION 1
22#endif
23
Baruch Siachc9eff6a2019-07-14 17:54:21 +030024#ifndef CONFIG_SPL_SATA_RAW_U_BOOT_SECTOR
25/* Dummy value to make the compiler happy */
26#define CONFIG_SPL_SATA_RAW_U_BOOT_SECTOR 0x100
27#endif
28
29static int spl_sata_load_image_raw(struct spl_image_info *spl_image,
Pali Rohárdda8f882022-01-14 14:31:38 +010030 struct spl_boot_device *bootdev,
Baruch Siachc9eff6a2019-07-14 17:54:21 +030031 struct blk_desc *stor_dev, unsigned long sector)
32{
Simon Glassbb7d3bb2022-09-06 20:26:52 -060033 struct legacy_img_hdr *header;
Baruch Siachc9eff6a2019-07-14 17:54:21 +030034 unsigned long count;
35 u32 image_size_sectors;
Pali Rohár11041672021-07-23 11:14:27 +020036 u32 image_offset_sectors;
37 u32 image_offset;
Baruch Siachc9eff6a2019-07-14 17:54:21 +030038 int ret;
39
40 header = spl_get_load_buffer(-sizeof(*header), stor_dev->blksz);
41 count = blk_dread(stor_dev, sector, 1, header);
42 if (count == 0)
43 return -EIO;
44
Pali Rohárdda8f882022-01-14 14:31:38 +010045 ret = spl_parse_image_header(spl_image, bootdev, header);
Baruch Siachc9eff6a2019-07-14 17:54:21 +030046 if (ret)
47 return ret;
48
49 image_size_sectors = DIV_ROUND_UP(spl_image->size, stor_dev->blksz);
Pali Rohár11041672021-07-23 11:14:27 +020050 image_offset_sectors = spl_image->offset / stor_dev->blksz;
51 image_offset = spl_image->offset % stor_dev->blksz;
52 count = blk_dread(stor_dev, sector + image_offset_sectors,
53 image_size_sectors,
Baruch Siachc9eff6a2019-07-14 17:54:21 +030054 (void *)spl_image->load_addr);
55 if (count != image_size_sectors)
56 return -EIO;
57
Pali Rohár11041672021-07-23 11:14:27 +020058 if (image_offset)
59 memmove((void *)spl_image->load_addr,
60 (void *)spl_image->load_addr + image_offset,
61 spl_image->size);
62
Baruch Siachc9eff6a2019-07-14 17:54:21 +030063 return 0;
64}
65
Simon Glassee306792016-09-24 18:20:13 -060066static int spl_sata_load_image(struct spl_image_info *spl_image,
67 struct spl_boot_device *bootdev)
Dan Murphy0c3ab852014-02-03 06:59:01 -060068{
Baruch Siach6b6d8622019-05-16 13:03:54 +030069 int err = 0;
Simon Glasse3394752016-02-29 15:25:34 -070070 struct blk_desc *stor_dev;
Dan Murphy0c3ab852014-02-03 06:59:01 -060071
Tom Rini6c7a9e02022-05-13 13:37:30 -040072 /* try to recognize storage devices immediately */
73 scsi_scan(false);
Simon Glassfada3f92022-09-17 09:00:09 -060074 stor_dev = blk_get_devnum_by_uclass_id(UCLASS_SCSI, 0);
Tom Rini6c7a9e02022-05-13 13:37:30 -040075 if (!stor_dev)
76 return -ENODEV;
Dan Murphy0c3ab852014-02-03 06:59:01 -060077
Tom Rinic2e78882021-10-30 23:03:48 -040078#if CONFIG_IS_ENABLED(OS_BOOT)
Simon Glass3eb382a2016-09-24 18:20:15 -060079 if (spl_start_uboot() ||
Pali Rohárdda8f882022-01-14 14:31:38 +010080 spl_load_image_fat_os(spl_image, bootdev, stor_dev,
Simon Glass3eb382a2016-09-24 18:20:15 -060081 CONFIG_SYS_SATA_FAT_BOOT_PARTITION))
Dan Murphy0c3ab852014-02-03 06:59:01 -060082#endif
Simon Glass3eb382a2016-09-24 18:20:15 -060083 {
Baruch Siach2f490e42019-05-16 13:03:55 +030084 err = -ENOSYS;
85
86 if (IS_ENABLED(CONFIG_SPL_FS_FAT)) {
Pali Rohárdda8f882022-01-14 14:31:38 +010087 err = spl_load_image_fat(spl_image, bootdev, stor_dev,
Simon Glass3eb382a2016-09-24 18:20:15 -060088 CONFIG_SYS_SATA_FAT_BOOT_PARTITION,
Baruch Siach2f490e42019-05-16 13:03:55 +030089 CONFIG_SPL_FS_LOAD_PAYLOAD_NAME);
Baruch Siachc9eff6a2019-07-14 17:54:21 +030090 } else if (IS_ENABLED(CONFIG_SPL_SATA_RAW_U_BOOT_USE_SECTOR)) {
Pali Rohárdda8f882022-01-14 14:31:38 +010091 err = spl_sata_load_image_raw(spl_image, bootdev, stor_dev,
Baruch Siachc9eff6a2019-07-14 17:54:21 +030092 CONFIG_SPL_SATA_RAW_U_BOOT_SECTOR);
Baruch Siach2f490e42019-05-16 13:03:55 +030093 }
Simon Glass3eb382a2016-09-24 18:20:15 -060094 }
Dan Murphy0c3ab852014-02-03 06:59:01 -060095 if (err) {
96 puts("Error loading sata device\n");
Nikita Kiryanov33eefe42015-11-08 17:11:49 +020097 return err;
Dan Murphy0c3ab852014-02-03 06:59:01 -060098 }
Nikita Kiryanov33eefe42015-11-08 17:11:49 +020099
100 return 0;
Dan Murphy0c3ab852014-02-03 06:59:01 -0600101}
Simon Glass4fc1f252016-11-30 15:30:50 -0700102SPL_LOAD_IMAGE_METHOD("SATA", 0, BOOT_DEVICE_SATA, spl_sata_load_image);