summaryrefslogtreecommitdiff
path: root/Kernel/Devices/GPTPartitionTable.cpp
blob: a4347bb716e5e8ccc12487a9b679ac1d61d30cd1 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
#include <AK/ByteBuffer.h>
#include <Kernel/Devices/GPTPartitionTable.h>

#define GPT_DEBUG

GPTPartitionTable::GPTPartitionTable(DiskDevice& device)
    : m_device(move(device))
{
}

GPTPartitionTable::~GPTPartitionTable()
{
}

const GPTPartitionHeader& GPTPartitionTable::header() const
{
    return *reinterpret_cast<const GPTPartitionHeader*>(m_cached_header);
}

bool GPTPartitionTable::initialize()
{
    if (!m_device->read_block(1, m_cached_header)) {
        return false;
    }

    auto& header = this->header();

#ifdef GPT_DEBUG
    kprintf("GPTPartitionTable::initialize: gpt_signature=%#x%x\n", header.sig[1], header.sig[0]);
#endif

    if (header.sig[0] != GPT_SIGNATURE && header.sig[1] != GPT_SIGNATURE2) {
        kprintf("GPTPartitionTable::initialize: bad GPT signature %#x%x\n", header.sig[1], header.sig[0]);
        return false;
    }

    return true;
}

RefPtr<DiskPartition> GPTPartitionTable::partition(unsigned index)
{
    ASSERT(index >= 1 && index <= 4294967294);

    auto& header = this->header();
    unsigned lba = header.partition_array_start_lba + (((index - 1) * header.partition_entry_size) / BytesPerSector);

    if (header.sig[0] != GPT_SIGNATURE) {
        kprintf("GPTPartitionTable::initialize: bad gpt signature - not initalized? %#x\n", header.sig);
        return nullptr;
    }

    u8 entries_per_sector = BytesPerSector / header.partition_entry_size;

    GPTPartitionEntry entries[entries_per_sector];
    this->m_device->read_blocks(lba, 1, (u8*)&entries);
    GPTPartitionEntry& entry = entries[((index - 1) % entries_per_sector)];

#ifdef GPT_DEBUG
    kprintf("GPTPartitionTable::partition %d\n", index);
    kprintf("GPTPartitionTable - offset = %d%d\n", entry.first_lba[1], entry.first_lba[0]);
#endif

    if (entry.first_lba[0] == 0x00) {
#ifdef GPT_DEBUG
        kprintf("GPTPartitionTable::partition: missing partition requested index=%d\n", index);
#endif

        return nullptr;
    }

#ifdef GPT_DEBUG
    kprintf("GPTPartitionTable::partition: found partition index=%d type=%x-%x-%x-%x\n", index, entry.partition_guid[3], entry.partition_guid[2], entry.partition_guid[1], entry.partition_guid[0]);
#endif
    return DiskPartition::create(m_device, entry.first_lba[0], entry.last_lba[0]);
}