Невозможно найти тип MBR - PullRequest
2 голосов
/ 13 марта 2012

У меня есть этот код, который является частью исходного кода проекта.Этот код находит тип MBR: GRUB или LILO и, соответственно, устанавливает флаг.

Удивительно, но в SLES 10-SP1 (SUSE Linux Enterprise Server) его невозможно определить./dev/sda1 мой своп./dev/sda2 - это место, где находится все /, включая MBR.

Тот же код работает для SLES11 и других.Здесь MBR_SIZE - от #defined до 0x1be.

int lnxfsGetBootType(int pNumber)
{
    int                 i, retval = -1, ccode;
    PartInfo            *p = &cpuParts[pNumber];
    char                buffer[SECTOR_SIZE];
    var64               offset = 0;

    isdLogFileOut(ZISD_LOG_DEVELOPER,"[lnxGBT]\n");
    if (getenv("ZENDEVICE") || gUtilPart == 1) {
        offset = p->pOffset;        // look at the partition BPB
    }

    //Now try to find the installed boot loader...
    lseek64(p->handle, (var64)offset, SEEK_SET);    // either MBR or BPB
    ccode = read(p->handle, buffer, SECTOR_SIZE);

    for (i=0; i<MBR_SIZE-4;i++) {
        if (strncmp(&buffer[i], "LILO", 4) == 0) {
            if (offset == 0){
                retval = FLAG_LNXFS_LILO;
                isdLogFileOut(ZISD_LOG_WARNING,"\tLILO MBR found on %s\n",p->header.deviceName);
            } else {
                retval = FLAG_LNXFS_LILO; // 10.31.06 _BPB;
                isdLogFileOut(ZISD_LOG_WARNING,"\tLILO BPB found on %s\n",p->header.deviceName);
            }
        }
        if (strncmp(&buffer[i], "GRUB", 4) == 0) {
            if (offset == 0){
                retval = FLAG_LNXFS_GRUB;
                isdLogFileOut(ZISD_LOG_WARNING,"\tGRUB MBR found on %s\n",p->header.deviceName);
            } else {
                retval = FLAG_LNXFS_GRUB; // 10.31.06 _BPB;
                isdLogFileOut(ZISD_LOG_WARNING,"\tGRUB BPB found on %s\n",p->header.deviceName);
            }
        }
    }
    if (retval == -1)  {
        isdLogFileOut(ZISD_LOG_WARNING,"\tLILO or GRUB mbr/bpb not found on %s\n",p->header.deviceName);
    }

    return retval;
}  // lnxfsGetBootType  

Here partinfo, is a struct of partition type:
//Data structure used internally by the image engine to store information about the 
//partitions.  It encapsulates the PartHeader struct, whcih is used to store partition
//information in image archives
typedef struct _PartInfo
{
    PartHeader  header;
    int         handle;         //file handle for reading/writing physical device
    var32       flags;          //Various flags as needed.  Defined above.
    var64       pOffset;        //offset to partition from start of physical device
    int         deviceNumber;   //index into 'devices' where this partition's 
                                // physical device is located
    int         archIndex;      //for restoring only.  Index into imgParts of the
                                // archive partition this physical partition is
                                // mapped to
    int         bytesWritten;   //track number of sectors written so the device-level
                                // cache can be flushed
    void        *info;          //partition-type-specific info struct

/* snip */

Тестирование выполняется с различными образами виртуальных дисков в VMWare.Я подтвердил, что диски отформатированы с использованием MBR, а не GPT.

1 Ответ

1 голос
/ 16 марта 2012

Я не уверен, что вы имеете в виду, когда говорите, что это не работает. Если вы считаете, что ваш код возвращает -1, не могли бы вы показать нам копию MBR? Вы можете использовать эту команду для захвата:

sudo dd if=/dev/sda bs=512 count=1 | xxd

Вы упоминаете, что ваш MBR на /dev/sda2. Это действительно очень необычно. Если вы имеете в виду, что именно здесь установлен загрузочный код, это совсем другое дело. MBR всегда хранится в первом секторе диска (при условии, что это MBR в формате DOS).

Я полагаю, что в некоторых случаях ошибки могут быть связаны с ошибкой поиска или коротким чтением. Я сделал несколько настроек, чтобы добавить обработку ошибок и немного упростить.

#define MBR_SIZE 0x1be

int lnxfsGetBootType(int pNumber)
{
    int                 retval = -1, ccode;
    PartInfo            *p = &cpuParts[pNumber];
    char                buffer[SECTOR_SIZE];
    off64_t  offset = 0;
    void *plilo, *pgrub;
    const char *what = "MBR";

    isdLogFileOut(ZISD_LOG_DEVELOPER,"[lnxGBT]\n");
    if (getenv("ZENDEVICE") || gUtilPart == 1) {
        offset = p->pOffset;        // look at the partition BPB
        what = "BPB";
    }

    // Now try to find the installed boot loader...
    if (lseek64(p->handle, offset, SEEK_SET) == -1) {
        isdLogFileOut(ZISD_LOG_ERROR,"\tFailed to seek to %s: %s\n", what, strerror(errno));
        return -1;
    }
    ccode = read(p->handle, buffer, SECTOR_SIZE);
    if (ccode != SECTOR_SIZE) {
            isdLogFileOut(ZISD_LOG_ERROR,"\tFailed to read BPB/MBR: %s\n",
                                strerror(errno));
            return -1;
    }
    plilo = memmem(buffer, ccode, "LILO", 4);
    pgrub = memmem(buffer, ccode, "GRUB", 4);
    if (plilo) {
        retval = FLAG_LNXFS_LILO;
        if (pgrub && pgrub < plilo)
            retval = FLAG_LNXFS_GRUB;
        }
    } else if (pgrub) {
        retval = FLAG_LNXFS_GRUB;
    }
    if (-1 == retval) {
        isdLogFileOut(ZISD_LOG_WARNING,"\tLILO or GRUB %s not found on %s\n", what, p->header.deviceName);
    } else {
        isdLogFileOut(ZISD_LOG_WARNING,"\t%s %s not found on %s\n", 
              (retval == FLAG_LNXFS_GRUB ? "GRUB" : "LILO"),
              what, p->header.deviceName);
    }
    return retval;
}  // lnxfsGetBootType  
...