Ignore:
File:
1 edited

Legend:

Unmodified
Added
Removed
  • uspace/lib/gpt/libgpt.c

    r44c4886 r9bdfde73  
    5151#include "libgpt.h"
    5252
    53 static int load_and_check_header(service_id_t handle, aoff64_t addr, size_t b_size, gpt_header_t * header);
     53static int load_and_check_header(service_id_t handle, aoff64_t addr, size_t b_size, gpt_header_t *header);
    5454static gpt_partitions_t * alloc_part_array(uint32_t num);
    55 static int extend_part_array(gpt_partitions_t * p);
    56 static int reduce_part_array(gpt_partitions_t * p);
     55static int extend_part_array(gpt_partitions_t *);
     56static int reduce_part_array(gpt_partitions_t *);
    5757static long long nearest_larger_int(double a);
    58 
     58static uint8_t get_byte(const char *);
     59static int check_overlap(gpt_part_t * p1, gpt_part_t * p2);
    5960
    6061/** Allocate memory for gpt label */
     
    179180
    180181        rc = block_init(EXCHANGE_ATOMIC, dev_handle, b_size);
    181         if (rc != EOK)
     182        if (rc != EOK && rc != EEXIST)
    182183                return rc;
    183184
     
    290291
    291292/** Write GPT and partitions to device
     293 * Note: also writes the header.
    292294 * @param label        label to write
    293295 * @param dev_handle   device to write the data to
     
    300302        size_t b_size;
    301303        uint32_t e_size = uint32_t_le2host(label->gpt->header->entry_size);
    302 
     304        size_t fill = label->parts->fill > GPT_MIN_PART_NUM ? label->parts->fill : GPT_MIN_PART_NUM;
     305       
    303306        label->gpt->header->pe_array_crc32 = compute_crc32(
    304307                                       (uint8_t *) label->parts->part_array,
    305                                        label->parts->fill * e_size);
     308                                       fill * e_size);
    306309       
    307310        /* comm_size of 4096 is ignored */
    308311        rc = block_init(EXCHANGE_ATOMIC, dev_handle, 4096);
    309         if (rc != EOK)
    310                 return rc;
    311 
     312        if (rc != EOK && rc != EEXIST)
     313                return rc;
     314       
    312315        rc = block_get_bsize(dev_handle, &b_size);
    313316        if (rc != EOK)
    314317                goto fail;
    315 
     318       
     319        aoff64_t n_blocks;
     320        rc = block_get_nblocks(dev_handle, &n_blocks);
     321        if (rc != EOK)
     322                goto fail;
     323       
     324        /* Write to backup GPT partition array location */
     325        //rc = block_write_direct(dev_handle, n_blocks - 1, GPT_HDR_BS, header->raw_data);
     326        if (rc != EOK)
     327                goto fail;
     328       
    316329        /* Write to main GPT partition array location */
    317330        rc = block_write_direct(dev_handle, uint64_t_le2host(label->gpt->header->entry_lba),
     
    320333        if (rc != EOK)
    321334                goto fail;
    322 
    323         aoff64_t n_blocks;
    324         rc = block_get_nblocks(dev_handle, &n_blocks);
    325         if (rc != EOK)
    326                 goto fail;
    327 
    328         /* Write to backup GPT partition array location */
    329         //rc = block_write_direct(dev_handle, n_blocks - 1, GPT_HDR_BS, header->raw_data);
    330         block_fini(dev_handle);
    331         if (rc != EOK)
    332                 goto fail;
    333 
    334 
     335       
    335336        return gpt_write_header(label, dev_handle);
    336337       
     
    347348 * This returns a memory block (zero-filled) and needs gpt_add_partition()
    348349 * to be called to insert it into a partition array.
    349  * Requires you to call gpt_free_partition after use.
     350 * Requires you to call gpt_free_partition afterwards.
    350351 */
    351352gpt_part_t * gpt_alloc_partition(void)
     
    367368 *
    368369 * Note: use either gpt_alloc_partition or gpt_get_partition.
    369  * This one return a pointer to a structure already inside the array, so
    370  * there's no need to call gpt_add_partition().
     370 * This one returns a pointer to the first empty structure already
     371 * inside the array, so don't call gpt_add_partition() afterwards.
    371372 * This is the one you will usually want.
    372373 */
    373374gpt_part_t * gpt_get_partition(gpt_label_t *label)
    374375{
    375         if (label->parts->fill == label->parts->arr_size) {
    376                 if (extend_part_array(label->parts) == -1)
    377                         return NULL;
    378         }
    379 
    380         return label->parts->part_array + label->parts->fill++;
     376        gpt_part_t *p;
     377       
     378        /* Find the first empty entry */
     379        do {
     380                if (label->parts->fill == label->parts->arr_size) {
     381                        if (extend_part_array(label->parts) == -1)
     382                                return NULL;
     383                }
     384               
     385                p = label->parts->part_array + label->parts->fill++;
     386               
     387        } while (gpt_get_part_type(p) != GPT_PTE_UNUSED);
     388       
     389        return p;
     390}
     391
     392/** Get partition already inside the label
     393 *
     394 * @param label   label to carrying the partition
     395 * @param idx     index of the partition
     396 *
     397 * @return        returns pointer to the partition
     398 *                or NULL when out of range
     399 *
     400 * Note: For new partitions use either gpt_alloc_partition or
     401 * gpt_get_partition unless you want a partition at a specific place.
     402 * This returns a pointer to a structure already inside the array,
     403 * so don't call gpt_add_partition() afterwards.
     404 * This function is handy when you want to change already existing
     405 * partition or to simply write somewhere in the middle. This works only
     406 * for indexes smaller than either 128 or the actual number of filled
     407 * entries.
     408 */
     409gpt_part_t * gpt_get_partition_at(gpt_label_t *label, size_t idx)
     410{
     411        return NULL;
     412       
     413        if (idx >= GPT_MIN_PART_NUM && idx >= label->parts->fill)
     414                return NULL;
     415       
     416        return label->parts->part_array + idx;
    381417}
    382418
     
    390426 * Note: for use with gpt_alloc_partition() only. You will get
    391427 * duplicates with gpt_get_partition().
     428 * Note: does not call gpt_free_partition()!
    392429 */
    393430int gpt_add_partition(gpt_label_t *label, gpt_part_t *partition)
     
    398435        }
    399436       
     437        /*FIXME:
     438         * Check dimensions and stuff! */
     439        gpt_part_foreach(label, p) {
     440                if (gpt_get_part_type(p) != GPT_PTE_UNUSED) {
     441                        if (check_overlap(partition, p))
     442                                return EINVAL;
     443                }
     444        }
     445       
    400446        memcpy(label->parts->part_array + label->parts->fill++,
    401447               partition, sizeof(gpt_part_t));
     448       
     449       
    402450       
    403451        return EOK;
     
    415463int gpt_remove_partition(gpt_label_t *label, size_t idx)
    416464{
    417         if (idx != label->parts->fill - 1) {
    418                 memmove(label->parts->part_array + idx,
    419                         label->parts->part_array + idx + 1,
    420                         (label->parts->fill - 1) * sizeof(gpt_entry_t));
    421                 label->parts->fill -= 1;
    422         }
    423        
    424         /* FIXME: This probably shouldn't be here, but instead
    425          * in reduce_part_array() or similar */
     465        if (idx >= label->parts->fill)
     466                return EINVAL;
     467       
     468        /* FIXME!
     469         * If we allow blank spots, we break the array. If we have more than
     470         * 128 partitions in the array and then remove something from
     471         * the first 128 partitions, we would forget to write the last one.*/
     472        memset(label->parts->part_array + idx, 0, sizeof(gpt_entry_t));
     473       
     474        label->parts->fill -= 1;
     475       
     476        /* FIXME! HOPEFULLY FIXED.
     477         * We cannot reduce the array so simply. We may have some partitions
     478         * there since we allow blank spots. */
     479        gpt_part_t * p;
    426480        if (label->parts->fill < (label->parts->arr_size / 2) - GPT_IGNORE_FILL_NUM) {
     481                for (p = gpt_get_partition_at(label, label->parts->arr_size / 2);
     482                     p < label->parts->part_array + label->parts->arr_size; ++p) {
     483                                if (gpt_get_part_type(p) != GPT_PTE_UNUSED)
     484                                        return EOK;
     485                }
     486               
    427487                if (reduce_part_array(label->parts) == ENOMEM)
    428488                        return ENOMEM;
     
    448508{
    449509        size_t i;
     510       
    450511        for (i = 0; gpt_ptypes[i].guid != NULL; i++) {
    451                 if (bcmp(p->part_type, gpt_ptypes[i].guid, 16) == 0) {
    452                         break;
    453                 }
    454         }
     512                if (p->part_type[3] == get_byte(gpt_ptypes[i].guid +0) &&
     513                        p->part_type[2] == get_byte(gpt_ptypes[i].guid +2) &&
     514                        p->part_type[1] == get_byte(gpt_ptypes[i].guid +4) &&
     515                        p->part_type[0] == get_byte(gpt_ptypes[i].guid +6) &&
     516                       
     517                        p->part_type[5] == get_byte(gpt_ptypes[i].guid +8) &&
     518                        p->part_type[4] == get_byte(gpt_ptypes[i].guid +10) &&
     519                       
     520                        p->part_type[7] == get_byte(gpt_ptypes[i].guid +12) &&
     521                        p->part_type[6] == get_byte(gpt_ptypes[i].guid +14) &&
     522                       
     523                        p->part_type[8] == get_byte(gpt_ptypes[i].guid +16) &&
     524                        p->part_type[9] == get_byte(gpt_ptypes[i].guid +18) &&
     525                        p->part_type[10] == get_byte(gpt_ptypes[i].guid +20) &&
     526                        p->part_type[11] == get_byte(gpt_ptypes[i].guid +22) &&
     527                        p->part_type[12] == get_byte(gpt_ptypes[i].guid +24) &&
     528                        p->part_type[13] == get_byte(gpt_ptypes[i].guid +26) &&
     529                        p->part_type[14] == get_byte(gpt_ptypes[i].guid +28) &&
     530                        p->part_type[15] == get_byte(gpt_ptypes[i].guid +30))
     531                                break;
     532        }
     533       
    455534        return i;
    456535}
     
    516595
    517596/** Copy partition name */
    518 void gpt_set_part_name(gpt_part_t * p, char * name[], size_t length)
     597void gpt_set_part_name(gpt_part_t *p, char *name, size_t length)
    519598{
    520599        if (length >= 72)
     
    645724}
    646725
    647 
    648 
    649 
    650 
    651 
     726static uint8_t get_byte(const char * c)
     727{
     728        uint8_t val = 0;
     729        char hex[3] = {*c, *(c+1), 0};
     730       
     731        errno = str_uint8_t(hex, NULL, 16, false, &val);
     732        return val;
     733}
     734
     735static int check_overlap(gpt_part_t * p1, gpt_part_t * p2)
     736{
     737        if (gpt_get_start_lba(p1) < gpt_get_start_lba(p2) && gpt_get_end_lba(p1) <= gpt_get_start_lba(p2)) {
     738                return 0;
     739        } else if (gpt_get_start_lba(p1) > gpt_get_start_lba(p2) && gpt_get_end_lba(p2) <= gpt_get_start_lba(p1)) {
     740                return 0;
     741        }
     742
     743        return 1;
     744}
     745
     746
Note: See TracChangeset for help on using the changeset viewer.