⭐ 欢迎来到虫虫下载站! | 📦 资源下载 📁 资源专辑 ℹ️ 关于我们
⭐ 虫虫下载站

📄 gpt.c

📁 Util-linux 软件包包含许多工具。其中比较重要的是加载、卸载、格式化、分区和管理硬盘驱动器
💻 C
📖 第 1 页 / 共 2 页
字号:
/*    gpt.[ch]    Copyright (C) 2000-2001 Dell Computer Corporation <Matt_Domsch@dell.com>     EFI GUID Partition Table handling    Per Intel EFI Specification v1.02    http://developer.intel.com/technology/efi/efi.htm    This program 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 2 of the License, or    (at your option) any later version.    This program 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 this program; if not, write to the Free Software    Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA  02111-1307  USA*/#define _FILE_OFFSET_BITS 64#include <stdio.h>#include <string.h>#include <stdlib.h>#include <inttypes.h>#include <sys/stat.h>#include <sys/ioctl.h>#include <fcntl.h>#include <unistd.h>#include <errno.h>#include <asm/byteorder.h>#include "crc32.h"#include "gpt.h"#include "partx.h"#define BLKGETLASTSECT  _IO(0x12,108)   /* get last sector of block device */#define BLKGETSIZE _IO(0x12,96)	        /* return device size */#define BLKSSZGET  _IO(0x12,104)	/* get block device sector size */#define BLKGETSIZE64 _IOR(0x12,114,sizeof(uint64_t))	/* return device size in bytes (u64 *arg) */struct blkdev_ioctl_param {        unsigned int block;        size_t content_length;        char * block_contents;};/** * efi_crc32() - EFI version of crc32 function * @buf: buffer to calculate crc32 of * @len - length of buf * * Description: Returns EFI-style CRC32 value for @buf *  * This function uses the little endian Ethernet polynomial * but seeds the function with ~0, and xor's with ~0 at the end. * Note, the EFI Specification, v1.02, has a reference to * Dr. Dobbs Journal, May 1994 (actually it's in May 1992). */static inline uint32_tefi_crc32(const void *buf, unsigned long len){	return (crc32(~0L, buf, len) ^ ~0L);}/** * is_pmbr_valid(): test Protective MBR for validity * @mbr: pointer to a legacy mbr structure * * Description: Returns 1 if PMBR is valid, 0 otherwise. * Validity depends on two things: *  1) MSDOS signature is in the last two bytes of the MBR *  2) One partition of type 0xEE is found */static intis_pmbr_valid(legacy_mbr *mbr){	int i, found = 0, signature = 0;	if (!mbr)		return 0;	signature = (__le16_to_cpu(mbr->signature) == MSDOS_MBR_SIGNATURE);	for (i = 0; signature && i < 4; i++) {		if (mbr->partition[i].sys_type ==                    EFI_PMBR_OSTYPE_EFI_GPT) {			found = 1;			break;		}	}	return (signature && found);}/************************************************************ * get_sector_size * Requires: *  - filedes is an open file descriptor, suitable for reading * Modifies: nothing * Returns: *  sector size, or 512. ************************************************************/static intget_sector_size(int filedes){	int rc, sector_size = 512;	rc = ioctl(filedes, BLKSSZGET, &sector_size);	if (rc)		sector_size = 512;	return sector_size;}/************************************************************ * _get_num_sectors * Requires: *  - filedes is an open file descriptor, suitable for reading * Modifies: nothing * Returns: *  Last LBA value on success  *  0 on error * * Try getting BLKGETSIZE64 and BLKSSZGET first, * then BLKGETSIZE if necessary. *  Kernels 2.4.15-2.4.18 and 2.5.0-2.5.3 have a broken BLKGETSIZE64 *  which returns the number of 512-byte sectors, not the size of *  the disk in bytes. Fixed in kernels 2.4.18-pre8 and 2.5.4-pre3. ************************************************************/static uint64_t_get_num_sectors(int filedes){	unsigned long sectors=0;	int rc;#if 0        uint64_t bytes=0; 	rc = ioctl(filedes, BLKGETSIZE64, &bytes);	if (!rc)		return bytes / get_sector_size(filedes);#endif        rc = ioctl(filedes, BLKGETSIZE, &sectors);        if (rc)                return 0;        	return sectors;}/************************************************************ * last_lba(): return number of last logical block of device *  * @fd *  * Description: returns Last LBA value on success, 0 on error. * Notes: The value st_blocks gives the size of the file *        in 512-byte blocks, which is OK if *        EFI_BLOCK_SIZE_SHIFT == 9. ************************************************************/static uint64_tlast_lba(int filedes){	int rc;	uint64_t sectors = 0;	struct stat s;	memset(&s, 0, sizeof (s));	rc = fstat(filedes, &s);	if (rc == -1) {		fprintf(stderr, "last_lba() could not stat: %s\n",			strerror(errno));		return 0;	}	if (S_ISBLK(s.st_mode)) {		sectors = _get_num_sectors(filedes);	} else {		fprintf(stderr,			"last_lba(): I don't know how to handle files with mode %x\n",			s.st_mode);		sectors = 1;	}	return sectors - 1;}static ssize_tread_lastoddsector(int fd, uint64_t lba, void *buffer, size_t count){        int rc;        struct blkdev_ioctl_param ioctl_param;        if (!buffer) return 0;         ioctl_param.block = 0; /* read the last sector */        ioctl_param.content_length = count;        ioctl_param.block_contents = buffer;        rc = ioctl(fd, BLKGETLASTSECT, &ioctl_param);        if (rc == -1) perror("read failed");        return !rc;}static ssize_tread_lba(int fd, uint64_t lba, void *buffer, size_t bytes){	int sector_size = get_sector_size(fd);	off_t offset = lba * sector_size;        ssize_t bytesread;	lseek(fd, offset, SEEK_SET);	bytesread = read(fd, buffer, bytes);        /* Kludge.  This is necessary to read/write the last           block of an odd-sized disk, until Linux 2.5.x kernel fixes.           This is only used by gpt.c, and only to read           one sector, so we don't have to be fancy.        */        if (!bytesread && !(last_lba(fd) & 1) && lba == last_lba(fd)) {                bytesread = read_lastoddsector(fd, lba, buffer, bytes);        }        return bytesread;}/** * alloc_read_gpt_entries(): reads partition entries from disk * @fd  is an open file descriptor to the whole disk * @gpt is a buffer into which the GPT will be put   * Description: Returns ptes on success,  NULL on error. * Allocates space for PTEs based on information found in @gpt. * Notes: remember to free pte when you're done! */static gpt_entry *alloc_read_gpt_entries(int fd, gpt_header * gpt){	gpt_entry *pte;        size_t count = __le32_to_cpu(gpt->num_partition_entries) *                __le32_to_cpu(gpt->sizeof_partition_entry);        if (!count) return NULL;	pte = (gpt_entry *)malloc(count);	if (!pte)		return NULL;	memset(pte, 0, count);	if (!read_lba(fd, __le64_to_cpu(gpt->partition_entry_lba), pte,                      count)) {		free(pte);		return NULL;	}	return pte;}/** * alloc_read_gpt_header(): Allocates GPT header, reads into it from disk * @fd  is an open file descriptor to the whole disk * @lba is the Logical Block Address of the partition table *  * Description: returns GPT header on success, NULL on error.   Allocates * and fills a GPT header starting at @ from @bdev. * Note: remember to free gpt when finished with it. */static gpt_header *alloc_read_gpt_header(int fd, uint64_t lba){	gpt_header *gpt;	gpt = (gpt_header *)	    malloc(sizeof (gpt_header));	if (!gpt)		return NULL;	memset(gpt, 0, sizeof (*gpt));	if (!read_lba(fd, lba, gpt, sizeof (gpt_header))) {		free(gpt);		return NULL;	}	return gpt;}/** * is_gpt_valid() - tests one GPT header and PTEs for validity * @fd  is an open file descriptor to the whole disk * @lba is the logical block address of the GPT header to test * @gpt is a GPT header ptr, filled on return. * @ptes is a PTEs ptr, filled on return. * * Description: returns 1 if valid,  0 on error. * If valid, returns pointers to newly allocated GPT header and PTEs. */static intis_gpt_valid(int fd, uint64_t lba,             gpt_header ** gpt, gpt_entry ** ptes){	int rc = 0;		/* default to not valid */	uint32_t crc, origcrc;	if (!gpt || !ptes)                return 0;	if (!(*gpt = alloc_read_gpt_header(fd, lba)))		return 0;	/* Check the GUID Partition Table signature */	if (__le64_to_cpu((*gpt)->signature) != GPT_HEADER_SIGNATURE) {

⌨️ 快捷键说明

复制代码 Ctrl + C
搜索代码 Ctrl + F
全屏模式 F11
切换主题 Ctrl + Shift + D
显示快捷键 ?
增大字号 Ctrl + =
减小字号 Ctrl + -