2012-03-13 2 views
2

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

Удивительно, что в 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.

+0

@sarnold: Нет, он протестирован с помощью VMware и отдельных виртуальных дисков. – kingsmasher1

+0

@sarnold: Нет, мы не используем GPT, это MBR. Просто подтвердил от команды. – kingsmasher1

+0

@sarnold: vmware4 disk image :) – kingsmasher1

ответ

1

Я не уверен, что вы имеете в виду, когда говорите, что это не работает. Если вы считаете, что ваш код возвращает -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 
+0

'/ dev/sda1' - это первый раздел на'/dev/sda' и будет * not * обычно иметь MBR (если только образ диска не виртуализирован в раздел). – wallyk

+0

@wallyk: Вы правы. –

+0

У меня есть MBR действительно на/dev/sda2, потому что я разделил и создал пространство подкачки на 1-м разделе. Кроме того, спасибо за исправленный код, вскоре вернется обратно после выполнения. Но странно эта проблема проявляется только на SLES 10-SP1 и SP2. На SP3 работает нормально – kingsmasher1

Смежные вопросы