]> git.ipfire.org Git - thirdparty/grub.git/commitdiff
uboot: Add the missing disk write operation support
authorCristian Ciocaltea <cristian.ciocaltea@gmail.com>
Tue, 22 Jan 2019 10:02:10 +0000 (12:02 +0200)
committerDaniel Kiper <daniel.kiper@oracle.com>
Tue, 22 Jan 2019 14:23:51 +0000 (15:23 +0100)
uboot_disk_write() is currently lacking the write support
to storage devices because, historically, those devices did not
implement block_write() in U-Boot.

The solution has been tested using a patched U-Boot loading
and booting GRUB in a QEMU vexpress-a9 environment.
The disk write operations were triggered with GRUB's save_env
command.

Signed-off-by: Cristian Ciocaltea <cristian.ciocaltea@gmail.com>
Reviewed-by: Daniel Kiper <daniel.kiper@oracle.com>
grub-core/disk/uboot/ubootdisk.c
grub-core/kern/uboot/uboot.c
include/grub/uboot/uboot.h

index a5ce07a99889c03fddc8b9a1a7b85ca385e0bfd1..584a9cbdf7bb5ca55f997342e0d84f2fe86025f4 100644 (file)
@@ -264,13 +264,23 @@ uboot_disk_read (struct grub_disk *disk,
 }
 
 static grub_err_t
-uboot_disk_write (struct grub_disk *disk __attribute__ ((unused)),
-                 grub_disk_addr_t sector __attribute__ ((unused)),
-                 grub_size_t size __attribute__ ((unused)),
-                 const char *buf __attribute__ ((unused)))
+uboot_disk_write (struct grub_disk *disk,
+                 grub_disk_addr_t offset, grub_size_t numblocks, const char *buf)
 {
-  return grub_error (GRUB_ERR_NOT_IMPLEMENTED_YET,
-                    "attempt to write (not supported)");
+  struct ubootdisk_data *d;
+  int retval;
+
+  d = disk->data;
+
+  retval = grub_uboot_dev_write (d->dev, buf, numblocks, offset);
+  grub_dprintf ("ubootdisk",
+               "retval=%d, numblocks=%d, sector=%llu\n",
+               retval, numblocks, (grub_uint64_t) offset);
+
+  if (retval != 0)
+    return grub_error (GRUB_ERR_IO, "U-Boot disk write error");
+
+  return GRUB_ERR_NONE;
 }
 
 static struct grub_disk_dev grub_ubootdisk_dev = {
index cf0168e62ddd26db50efd2351499fa7f74fce26c..be4816fe6f592bb2adc841aca1009dd69218c273 100644 (file)
@@ -243,6 +243,22 @@ grub_uboot_dev_read (struct device_info *dev, void *buf, grub_size_t blocks,
   return retval;
 }
 
+int
+grub_uboot_dev_write (struct device_info *dev, const void *buf,
+                     grub_size_t blocks, grub_uint32_t start)
+{
+  int retval;
+
+  if (!OPEN_DEV (dev))
+    return -1;
+
+  if (!grub_uboot_syscall (API_DEV_WRITE, &retval, dev, buf,
+                          &blocks, &start))
+    return -1;
+
+  return retval;
+}
+
 int
 grub_uboot_dev_recv (struct device_info *dev, void *buf,
                     int size, int *real_size)
index c122de6abf68a8ecff6f62c230e93ec005551a3e..194130345a72fda9b2e6bd572dbab59b1792c141 100644 (file)
@@ -72,7 +72,8 @@ int EXPORT_FUNC (grub_uboot_dev_enum) (void);
 struct device_info * EXPORT_FUNC (grub_uboot_dev_get) (int index);
 int EXPORT_FUNC (grub_uboot_dev_open) (struct device_info *dev);
 int EXPORT_FUNC (grub_uboot_dev_close) (struct device_info *dev);
-int grub_uboot_dev_write (struct device_info *dev, void *buf, int *len);
+int grub_uboot_dev_write (struct device_info *dev, const void *buf,
+                         grub_size_t blocks, grub_uint32_t start);
 int grub_uboot_dev_read (struct device_info *dev, void *buf, grub_size_t blocks,
                         grub_uint32_t start, grub_size_t * real_blocks);
 int EXPORT_FUNC (grub_uboot_dev_recv) (struct device_info *dev, void *buf,