|  | // SPDX-License-Identifier: GPL-2.0+ | 
|  | /* | 
|  | *  Software partition device (UCLASS_PARTITION) | 
|  | * | 
|  | *  Copyright (c) 2021 Linaro Limited | 
|  | *			Author: AKASHI Takahiro | 
|  | */ | 
|  |  | 
|  | #define LOG_CATEGORY UCLASS_PARTITION | 
|  |  | 
|  | #include <blk.h> | 
|  | #include <dm.h> | 
|  | #include <log.h> | 
|  | #include <part.h> | 
|  | #include <vsprintf.h> | 
|  | #include <dm/device-internal.h> | 
|  | #include <dm/lists.h> | 
|  |  | 
|  | /** | 
|  | * disk_blk_part_validate() - Check whether access to partition is within limits | 
|  | * | 
|  | * @dev: Device (partition udevice) | 
|  | * @start: Start block for the access(from start of partition) | 
|  | * @blkcnt: Number of blocks to access (within the partition) | 
|  | * @return 0 on valid block range, or -ve on error. | 
|  | */ | 
|  | static int disk_blk_part_validate(struct udevice *dev, lbaint_t start, lbaint_t blkcnt) | 
|  | { | 
|  | struct disk_part *part = dev_get_uclass_plat(dev); | 
|  |  | 
|  | if (device_get_uclass_id(dev) != UCLASS_PARTITION) | 
|  | return -ENOSYS; | 
|  |  | 
|  | if (start >= part->gpt_part_info.size) | 
|  | return -E2BIG; | 
|  |  | 
|  | if ((start + blkcnt) > part->gpt_part_info.size) | 
|  | return -ERANGE; | 
|  |  | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | /** | 
|  | * disk_blk_part_offset() - Compute offset from start of block device | 
|  | * | 
|  | * @dev: Device (partition udevice) | 
|  | * @start: Start block for the access (from start of partition) | 
|  | * @return Start block for the access (from start of block device) | 
|  | */ | 
|  | static lbaint_t disk_blk_part_offset(struct udevice *dev, lbaint_t start) | 
|  | { | 
|  | struct disk_part *part = dev_get_uclass_plat(dev); | 
|  |  | 
|  | return start + part->gpt_part_info.start; | 
|  | } | 
|  |  | 
|  | /* | 
|  | * BLOCK IO APIs | 
|  | */ | 
|  | /** | 
|  | * disk_blk_read() - Read from a block device partition | 
|  | * | 
|  | * @dev: Device to read from (partition udevice) | 
|  | * @start: Start block for the read (from start of partition) | 
|  | * @blkcnt: Number of blocks to read (within the partition) | 
|  | * @buffer: Place to put the data | 
|  | * @return number of blocks read (which may be less than @blkcnt), | 
|  | * or -ve on error. This never returns 0 unless @blkcnt is 0 | 
|  | */ | 
|  | unsigned long disk_blk_read(struct udevice *dev, lbaint_t start, | 
|  | lbaint_t blkcnt, void *buffer) | 
|  | { | 
|  | int ret = disk_blk_part_validate(dev, start, blkcnt); | 
|  |  | 
|  | if (ret) | 
|  | return ret; | 
|  |  | 
|  | return blk_read(dev_get_parent(dev), disk_blk_part_offset(dev, start), | 
|  | blkcnt, buffer); | 
|  | } | 
|  |  | 
|  | /** | 
|  | * disk_blk_write() - Write to a block device | 
|  | * | 
|  | * @dev: Device to write to (partition udevice) | 
|  | * @start: Start block for the write (from start of partition) | 
|  | * @blkcnt: Number of blocks to write (within the partition) | 
|  | * @buffer: Data to write | 
|  | * @return number of blocks written (which may be less than @blkcnt), | 
|  | * or -ve on error. This never returns 0 unless @blkcnt is 0 | 
|  | */ | 
|  | unsigned long disk_blk_write(struct udevice *dev, lbaint_t start, | 
|  | lbaint_t blkcnt, const void *buffer) | 
|  | { | 
|  | int ret = disk_blk_part_validate(dev, start, blkcnt); | 
|  |  | 
|  | if (ret) | 
|  | return ret; | 
|  |  | 
|  | return blk_write(dev_get_parent(dev), disk_blk_part_offset(dev, start), | 
|  | blkcnt, buffer); | 
|  | } | 
|  |  | 
|  | /** | 
|  | * disk_blk_erase() - Erase part of a block device | 
|  | * | 
|  | * @dev: Device to erase (partition udevice) | 
|  | * @start: Start block for the erase (from start of partition) | 
|  | * @blkcnt: Number of blocks to erase (within the partition) | 
|  | * @return number of blocks erased (which may be less than @blkcnt), | 
|  | * or -ve on error. This never returns 0 unless @blkcnt is 0 | 
|  | */ | 
|  | unsigned long disk_blk_erase(struct udevice *dev, lbaint_t start, | 
|  | lbaint_t blkcnt) | 
|  | { | 
|  | int ret = disk_blk_part_validate(dev, start, blkcnt); | 
|  |  | 
|  | if (ret) | 
|  | return ret; | 
|  |  | 
|  | return blk_erase(dev_get_parent(dev), disk_blk_part_offset(dev, start), | 
|  | blkcnt); | 
|  | } | 
|  |  | 
|  | UCLASS_DRIVER(partition) = { | 
|  | .id		= UCLASS_PARTITION, | 
|  | .per_device_plat_auto	= sizeof(struct disk_part), | 
|  | .name		= "partition", | 
|  | }; | 
|  |  | 
|  | static const struct blk_ops blk_part_ops = { | 
|  | .read	= disk_blk_read, | 
|  | .write	= disk_blk_write, | 
|  | .erase	= disk_blk_erase, | 
|  | }; | 
|  |  | 
|  | U_BOOT_DRIVER(blk_partition) = { | 
|  | .name		= "blk_partition", | 
|  | .id		= UCLASS_PARTITION, | 
|  | .ops		= &blk_part_ops, | 
|  | }; |