mirror of
				https://git.proxmox.com/git/grub2
				synced 2025-11-04 10:24:12 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			150 lines
		
	
	
		
			3.5 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			150 lines
		
	
	
		
			3.5 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
/* fs_uuid.c - Access disks by their filesystem UUID.  */
 | 
						|
/*
 | 
						|
 *  GRUB  --  GRand Unified Bootloader
 | 
						|
 *  Copyright (C) 2007,2008  Free Software Foundation, Inc.
 | 
						|
 *
 | 
						|
 *  GRUB is free software: you can redistribute it and/or modify
 | 
						|
 *  it under the terms of the GNU General Public License as published by
 | 
						|
 *  the Free Software Foundation, either version 3 of the License, or
 | 
						|
 *  (at your option) any later version.
 | 
						|
 *
 | 
						|
 *  GRUB is distributed in the hope that it will be useful,
 | 
						|
 *  but WITHOUT ANY WARRANTY; without even the implied warranty of
 | 
						|
 *  MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 | 
						|
 *  GNU General Public License for more details.
 | 
						|
 *
 | 
						|
 *  You should have received a copy of the GNU General Public License
 | 
						|
 *  along with GRUB.  If not, see <http://www.gnu.org/licenses/>.
 | 
						|
 */
 | 
						|
 | 
						|
#include <grub/disk.h>
 | 
						|
#include <grub/dl.h>
 | 
						|
#include <grub/kernel.h>
 | 
						|
#include <grub/misc.h>
 | 
						|
#include <grub/mm.h>
 | 
						|
#include <grub/types.h>
 | 
						|
 | 
						|
#include <grub/fs.h>
 | 
						|
#include <grub/partition.h>
 | 
						|
 | 
						|
static grub_device_t
 | 
						|
search_fs_uuid (const char *key, unsigned long *count)
 | 
						|
{
 | 
						|
  *count = 0;
 | 
						|
  grub_device_t ret = NULL;
 | 
						|
 | 
						|
  auto int iterate_device (const char *name);
 | 
						|
  int iterate_device (const char *name)
 | 
						|
    {
 | 
						|
      grub_device_t dev;
 | 
						|
 | 
						|
      dev = grub_device_open (name);
 | 
						|
      if (dev)
 | 
						|
	{
 | 
						|
	  grub_fs_t fs;
 | 
						|
 | 
						|
	  fs = grub_fs_probe (dev);
 | 
						|
	  if (fs && fs->uuid)
 | 
						|
	    {
 | 
						|
	      char *uuid;
 | 
						|
 | 
						|
	      (fs->uuid) (dev, &uuid);
 | 
						|
	      if (grub_errno == GRUB_ERR_NONE && uuid)
 | 
						|
		{
 | 
						|
		  (*count)++;
 | 
						|
 | 
						|
		  if (grub_strcasecmp (uuid, key) == 0)
 | 
						|
		    {
 | 
						|
		      ret = dev;
 | 
						|
		      grub_free (uuid);
 | 
						|
		      return 1;
 | 
						|
		    }
 | 
						|
		  grub_free (uuid);
 | 
						|
		}
 | 
						|
	    }
 | 
						|
 | 
						|
	  grub_device_close (dev);
 | 
						|
	}
 | 
						|
 | 
						|
      grub_errno = GRUB_ERR_NONE;
 | 
						|
      return 0;
 | 
						|
    }
 | 
						|
 | 
						|
  grub_device_iterate (iterate_device);
 | 
						|
 | 
						|
  return ret;
 | 
						|
}
 | 
						|
 | 
						|
static grub_err_t
 | 
						|
grub_fs_uuid_open (const char *name, grub_disk_t disk)
 | 
						|
{
 | 
						|
  grub_device_t dev;
 | 
						|
 | 
						|
  if (grub_strncmp (name, "UUID=", sizeof ("UUID=")-1))
 | 
						|
    return grub_error (GRUB_ERR_UNKNOWN_DEVICE, "not a UUID virtual volume");
 | 
						|
 | 
						|
  dev = search_fs_uuid (name + sizeof ("UUID=") - 1, &disk->id);
 | 
						|
  if (! dev)
 | 
						|
    return grub_error (GRUB_ERR_UNKNOWN_DEVICE, "no matching UUID found");
 | 
						|
 | 
						|
  disk->total_sectors = dev->disk->total_sectors;
 | 
						|
  disk->has_partitions = 0;
 | 
						|
  if (dev->disk->partition)
 | 
						|
    {
 | 
						|
      disk->partition = grub_malloc (sizeof (*disk->partition));
 | 
						|
      if (disk->partition)
 | 
						|
	grub_memcpy (disk->partition, dev->disk->partition,
 | 
						|
		     sizeof (*disk->partition));
 | 
						|
    }
 | 
						|
  else
 | 
						|
    disk->partition = NULL;
 | 
						|
 | 
						|
  disk->data = dev;
 | 
						|
 | 
						|
  return GRUB_ERR_NONE;
 | 
						|
}
 | 
						|
 | 
						|
static void
 | 
						|
grub_fs_uuid_close (grub_disk_t disk __attribute((unused)))
 | 
						|
{
 | 
						|
  grub_device_t parent = disk->data;
 | 
						|
  grub_device_close (parent);
 | 
						|
}
 | 
						|
 | 
						|
static grub_err_t
 | 
						|
grub_fs_uuid_read (grub_disk_t disk, grub_disk_addr_t sector,
 | 
						|
		   grub_size_t size, char *buf)
 | 
						|
{
 | 
						|
  grub_device_t parent = disk->data;
 | 
						|
  return parent->disk->dev->read (parent->disk, sector, size, buf);
 | 
						|
}
 | 
						|
 | 
						|
static grub_err_t
 | 
						|
grub_fs_uuid_write (grub_disk_t disk, grub_disk_addr_t sector,
 | 
						|
		    grub_size_t size, const char *buf)
 | 
						|
{
 | 
						|
  grub_device_t parent = disk->data;
 | 
						|
  return parent->disk->dev->write (parent->disk, sector, size, buf);
 | 
						|
}
 | 
						|
 | 
						|
static struct grub_disk_dev grub_fs_uuid_dev =
 | 
						|
  {
 | 
						|
    .name = "fs_uuid",
 | 
						|
    .id = GRUB_DISK_DEVICE_UUID_ID,
 | 
						|
    .open = grub_fs_uuid_open,
 | 
						|
    .close = grub_fs_uuid_close,
 | 
						|
    .read = grub_fs_uuid_read,
 | 
						|
    .write = grub_fs_uuid_write,
 | 
						|
    .next = 0
 | 
						|
  };
 | 
						|
 | 
						|
GRUB_MOD_INIT(fs_uuid)
 | 
						|
{
 | 
						|
  grub_disk_dev_register (&grub_fs_uuid_dev);
 | 
						|
}
 | 
						|
 | 
						|
GRUB_MOD_FINI(fs_uuid)
 | 
						|
{
 | 
						|
  grub_disk_dev_unregister (&grub_fs_uuid_dev);
 | 
						|
}
 |