File indexing completed on 2025-05-11 08:24:14
0001
0002
0003
0004
0005
0006
0007
0008
0009
0010
0011
0012
0013
0014
0015
0016
0017
0018
0019
0020
0021
0022
0023
0024
0025
0026
0027
0028
0029
0030
0031
0032
0033
0034
0035
0036
0037
0038
0039
0040 #ifdef HAVE_CONFIG_H
0041 #include "config.h"
0042 #endif
0043
0044 #include <string.h>
0045
0046 #include <rtems/ide_part_table.h>
0047
0048
0049
0050
0051
0052
0053
0054
0055
0056
0057
0058
0059
0060
0061
0062
0063
0064
0065
0066
0067 static rtems_status_code
0068 get_sector(int fd,
0069 uint32_t sector_num,
0070 rtems_sector_data_t **sector)
0071 {
0072 rtems_sector_data_t *s;
0073 ssize_t n;
0074 off_t off;
0075 off_t new_off;
0076
0077 if (sector == NULL)
0078 {
0079 return RTEMS_INTERNAL_ERROR;
0080 }
0081
0082 off = sector_num * RTEMS_IDE_SECTOR_SIZE;
0083 new_off = lseek(fd, off, SEEK_SET);
0084 if (new_off != off) {
0085 return RTEMS_IO_ERROR;
0086 }
0087
0088 s = (rtems_sector_data_t *) malloc(sizeof(rtems_sector_data_t) + RTEMS_IDE_SECTOR_SIZE);
0089 if (s == NULL)
0090 {
0091 return RTEMS_NO_MEMORY;
0092 }
0093
0094 n = read(fd, s->data, RTEMS_IDE_SECTOR_SIZE);
0095 if (n != RTEMS_IDE_SECTOR_SIZE)
0096 {
0097 free(s);
0098 return RTEMS_IO_ERROR;
0099 }
0100
0101 s->sector_num = sector_num;
0102
0103 *sector = s;
0104
0105 return RTEMS_SUCCESSFUL;
0106 }
0107
0108
0109
0110
0111
0112
0113
0114
0115
0116
0117
0118
0119 static bool
0120 msdos_signature_check (rtems_sector_data_t *sector)
0121 {
0122 uint8_t *p = sector->data + RTEMS_IDE_PARTITION_MSDOS_SIGNATURE_OFFSET;
0123
0124 return ((p[0] == RTEMS_IDE_PARTITION_MSDOS_SIGNATURE_DATA1) &&
0125 (p[1] == RTEMS_IDE_PARTITION_MSDOS_SIGNATURE_DATA2));
0126 }
0127
0128
0129
0130
0131
0132
0133
0134
0135
0136
0137
0138
0139 static bool
0140 is_extended(uint8_t type)
0141 {
0142 return ((type == EXTENDED_PARTITION) || (type == LINUX_EXTENDED));
0143 }
0144
0145
0146
0147
0148
0149
0150
0151
0152
0153
0154
0155 static bool
0156 is_fat_partition(uint8_t type)
0157 {
0158 static const uint8_t fat_part_types[] = {
0159 DOS_FAT12_PARTITION,DOS_FAT16_PARTITION,
0160 DOS_P32MB_PARTITION,
0161 FAT32_PARTITION ,FAT32_LBA_PARTITION,
0162 FAT16_LBA_PARTITION
0163 };
0164
0165 return (NULL != memchr(fat_part_types,type,sizeof(fat_part_types)));
0166 }
0167
0168
0169
0170
0171
0172
0173
0174
0175
0176
0177
0178
0179
0180
0181
0182
0183 static rtems_status_code
0184 data_to_part_desc(uint8_t *data, rtems_part_desc_t **new_part_desc)
0185 {
0186 rtems_part_desc_t *part_desc;
0187 uint32_t temp;
0188
0189 if (new_part_desc == NULL)
0190 {
0191 return RTEMS_INTERNAL_ERROR;
0192 }
0193
0194 *new_part_desc = NULL;
0195
0196 if ((part_desc = calloc(1, sizeof(rtems_part_desc_t))) == NULL)
0197 {
0198 return RTEMS_NO_MEMORY;
0199 }
0200
0201 part_desc->bootable = *(data + RTEMS_IDE_PARTITION_BOOTABLE_OFFSET);
0202 part_desc->sys_type = *(data + RTEMS_IDE_PARTITION_SYS_TYPE_OFFSET);
0203
0204
0205
0206
0207 memcpy(&temp, data + RTEMS_IDE_PARTITION_START_OFFSET, sizeof(uint32_t));
0208 part_desc->start = LE_TO_CPU_U32(temp);
0209
0210 memcpy(&temp, data + RTEMS_IDE_PARTITION_SIZE_OFFSET, sizeof(uint32_t));
0211 part_desc->size = LE_TO_CPU_U32(temp);
0212
0213
0214
0215
0216
0217
0218
0219 if (is_extended(part_desc->sys_type) ||
0220 ((is_fat_partition(part_desc->sys_type)) && (part_desc->size != 0))) {
0221 *new_part_desc = part_desc;
0222 }
0223 else {
0224
0225 free(part_desc);
0226 }
0227 return RTEMS_SUCCESSFUL;
0228 }
0229
0230
0231
0232
0233
0234
0235
0236
0237
0238
0239
0240
0241
0242
0243
0244
0245
0246
0247 static rtems_status_code
0248 read_extended_partition(int fd, uint32_t start, rtems_part_desc_t *ext_part)
0249 {
0250 int i;
0251 rtems_sector_data_t *sector = NULL;
0252 uint32_t here;
0253 uint8_t *data;
0254 rtems_part_desc_t *new_part_desc;
0255 rtems_status_code rc;
0256
0257 if ((ext_part == NULL) || (ext_part->disk_desc == NULL))
0258 {
0259 return RTEMS_INTERNAL_ERROR;
0260 }
0261
0262
0263 here = ext_part->start;
0264
0265
0266
0267 rc = get_sector(fd, here, §or);
0268 if (rc != RTEMS_SUCCESSFUL)
0269 {
0270 if (sector)
0271 free(sector);
0272 return rc;
0273 }
0274
0275 if (!msdos_signature_check(sector))
0276 {
0277 free(sector);
0278 return RTEMS_INTERNAL_ERROR;
0279 }
0280
0281
0282
0283 data = sector->data + RTEMS_IDE_PARTITION_TABLE_OFFSET;
0284
0285 for (i = 0; i < RTEMS_IDE_PARTITION_MAX_SUB_PARTITION_NUMBER; i++)
0286 {
0287
0288
0289
0290 rc = data_to_part_desc(data, &new_part_desc);
0291 if (rc != RTEMS_SUCCESSFUL)
0292 {
0293 free(sector);
0294 return rc;
0295 }
0296
0297 if (new_part_desc == NULL)
0298 {
0299 data += RTEMS_IDE_PARTITION_DESCRIPTOR_SIZE;
0300 continue;
0301 }
0302
0303 ext_part->sub_part[i] = new_part_desc;
0304 new_part_desc->ext_part = ext_part;
0305 new_part_desc->disk_desc = ext_part->disk_desc;
0306
0307 if (is_extended(new_part_desc->sys_type))
0308 {
0309 new_part_desc->log_id = EMPTY_PARTITION;
0310 new_part_desc->start += start;
0311 read_extended_partition(fd, start, new_part_desc);
0312 }
0313 else
0314 {
0315 rtems_disk_desc_t *disk_desc = new_part_desc->disk_desc;
0316 disk_desc->partitions[disk_desc->last_log_id] = new_part_desc;
0317 new_part_desc->log_id = ++disk_desc->last_log_id;
0318 new_part_desc->start += here;
0319 new_part_desc->end = new_part_desc->start + new_part_desc->size - 1;
0320 }
0321 data += RTEMS_IDE_PARTITION_DESCRIPTOR_SIZE;
0322 }
0323
0324 free(sector);
0325
0326 return RTEMS_SUCCESSFUL;
0327 }
0328
0329
0330
0331
0332
0333
0334
0335
0336
0337
0338
0339
0340
0341
0342 static rtems_status_code
0343 read_mbr(int fd, rtems_disk_desc_t *disk_desc)
0344 {
0345 int part_num;
0346 rtems_sector_data_t *sector = NULL;
0347 rtems_part_desc_t *part_desc;
0348 uint8_t *data;
0349 rtems_status_code rc;
0350
0351
0352 rc = get_sector(fd, 0, §or);
0353 if (rc != RTEMS_SUCCESSFUL)
0354 {
0355 if (sector)
0356 free(sector);
0357 return rc;
0358 }
0359
0360
0361 if (!msdos_signature_check(sector))
0362 {
0363 free(sector);
0364 return RTEMS_INTERNAL_ERROR;
0365 }
0366
0367
0368
0369 data = sector->data + RTEMS_IDE_PARTITION_TABLE_OFFSET;
0370
0371 for (part_num = 0;
0372 part_num < RTEMS_IDE_PARTITION_MAX_SUB_PARTITION_NUMBER;
0373 part_num++)
0374 {
0375 rc = data_to_part_desc(data, &part_desc);
0376 if (rc != RTEMS_SUCCESSFUL)
0377 {
0378 free(sector);
0379 return rc;
0380 }
0381
0382 if (part_desc != NULL)
0383 {
0384 part_desc->log_id = part_num + 1;
0385 part_desc->disk_desc = disk_desc;
0386 part_desc->end = part_desc->start + part_desc->size - 1;
0387 disk_desc->partitions[part_num] = part_desc;
0388 }
0389 else
0390 {
0391 disk_desc->partitions[part_num] = NULL;
0392 }
0393
0394 data += RTEMS_IDE_PARTITION_DESCRIPTOR_SIZE;
0395 }
0396
0397 free(sector);
0398
0399 disk_desc->last_log_id = RTEMS_IDE_PARTITION_MAX_SUB_PARTITION_NUMBER;
0400
0401
0402
0403 for (part_num = 0;
0404 part_num < RTEMS_IDE_PARTITION_MAX_SUB_PARTITION_NUMBER;
0405 part_num++)
0406 {
0407 part_desc = disk_desc->partitions[part_num];
0408 if (part_desc != NULL && is_extended(part_desc->sys_type))
0409 {
0410 read_extended_partition(fd, part_desc->start, part_desc);
0411 free(part_desc);
0412 disk_desc->partitions[part_num] = NULL;
0413 }
0414 }
0415
0416 return RTEMS_SUCCESSFUL;
0417 }
0418
0419
0420
0421
0422
0423
0424
0425
0426
0427
0428
0429
0430 static void
0431 partition_free(rtems_part_desc_t *part_desc)
0432 {
0433 int part_num;
0434
0435 if (part_desc == NULL)
0436 return;
0437
0438 if (is_extended(part_desc->sys_type))
0439 {
0440 for (part_num = 0;
0441 part_num < RTEMS_IDE_PARTITION_MAX_SUB_PARTITION_NUMBER;
0442 part_num++)
0443 {
0444 partition_free(part_desc->sub_part[part_num]);
0445 }
0446 }
0447
0448 free(part_desc);
0449 }
0450
0451
0452
0453
0454
0455
0456
0457
0458
0459
0460
0461 static void
0462 partition_table_free(rtems_disk_desc_t *disk_desc)
0463 {
0464 int part_num;
0465
0466 for (part_num = 0;
0467 part_num < RTEMS_IDE_PARTITION_MAX_SUB_PARTITION_NUMBER;
0468 part_num++)
0469 {
0470 partition_free(disk_desc->partitions[part_num]);
0471 }
0472
0473 free(disk_desc);
0474 }
0475
0476
0477
0478
0479
0480
0481
0482
0483
0484
0485
0486
0487
0488
0489 static rtems_status_code
0490 partition_table_get(const char *dev_name, rtems_disk_desc_t *disk_desc)
0491 {
0492 struct stat dev_stat;
0493 rtems_status_code rc;
0494 int fd;
0495
0496 fd = open(dev_name, O_RDONLY);
0497 if (fd < 0)
0498 {
0499 return RTEMS_INTERNAL_ERROR;
0500 }
0501
0502 rc = fstat(fd, &dev_stat);
0503 if (rc != RTEMS_SUCCESSFUL)
0504 {
0505 close(fd);
0506 return RTEMS_INTERNAL_ERROR;
0507 }
0508
0509 strncpy (disk_desc->dev_name, dev_name, 15);
0510 disk_desc->sector_size = (dev_stat.st_blksize) ? dev_stat.st_blksize :
0511 RTEMS_IDE_SECTOR_SIZE;
0512
0513 rc = read_mbr(fd, disk_desc);
0514
0515 close(fd);
0516
0517 return rc;
0518 }
0519
0520
0521
0522
0523
0524
0525
0526
0527
0528
0529
0530 void
0531 rtems_ide_part_table_free(rtems_disk_desc_t *disk_desc)
0532 {
0533 partition_table_free( disk_desc );
0534 }
0535
0536
0537
0538
0539
0540
0541
0542
0543
0544
0545
0546
0547
0548
0549 rtems_status_code
0550 rtems_ide_part_table_get(const char *dev_name, rtems_disk_desc_t *disk_desc)
0551 {
0552 return partition_table_get( dev_name, disk_desc );
0553 }
0554
0555
0556
0557
0558
0559
0560
0561
0562
0563
0564
0565
0566
0567
0568 rtems_status_code
0569 rtems_ide_part_table_initialize(const char *dev_name)
0570 {
0571 int part_num;
0572 rtems_disk_desc_t *disk_desc;
0573 rtems_status_code rc;
0574 rtems_part_desc_t *part_desc;
0575
0576
0577 char name[RTEMS_IDE_PARTITION_DEV_NAME_LENGTH_MAX];
0578
0579 disk_desc = (rtems_disk_desc_t *) calloc(1, sizeof(rtems_disk_desc_t));
0580 if (disk_desc == NULL)
0581 {
0582 return RTEMS_NO_MEMORY;
0583 }
0584
0585
0586 rc = partition_table_get(dev_name, disk_desc);
0587 if (rc != RTEMS_SUCCESSFUL)
0588 {
0589 free(disk_desc);
0590 return rc;
0591 }
0592
0593
0594 for (part_num = 0; part_num < disk_desc->last_log_id; part_num++)
0595 {
0596 sprintf(name, "%s%d", dev_name, part_num + 1);
0597
0598 part_desc = disk_desc->partitions[part_num];
0599 if (part_desc == NULL)
0600 {
0601 continue;
0602 }
0603
0604 rc = rtems_blkdev_create_partition(name, dev_name, part_desc->start,
0605 part_desc->size);
0606 if (rc != RTEMS_SUCCESSFUL)
0607 {
0608 fprintf(stdout,"Cannot create device %s, error code %d\n", name, rc);
0609 continue;
0610 }
0611 }
0612
0613 partition_table_free(disk_desc);
0614
0615 return RTEMS_SUCCESSFUL;
0616 }