2013-09-18 00:46:58 +02:00
|
|
|
from abstract import AbstractPartitionMap
|
|
|
|
from ..partitions.gpt import GPTPartition
|
|
|
|
from ..partitions.gpt_swap import GPTSwapPartition
|
2015-01-01 21:09:16 +01:00
|
|
|
from ..partitions.gap import PartitionGap
|
2014-03-23 23:12:07 +01:00
|
|
|
from bootstrapvz.common.tools import log_check_call
|
2013-09-18 00:46:58 +02:00
|
|
|
|
|
|
|
|
|
|
|
class GPTPartitionMap(AbstractPartitionMap):
|
2014-03-23 16:04:03 +01:00
|
|
|
"""Represents a GPT partition map
|
|
|
|
"""
|
2013-09-18 00:46:58 +02:00
|
|
|
|
2015-01-01 21:09:16 +01:00
|
|
|
def __init__(self, data, sector_size, bootloader):
|
2014-03-23 16:04:03 +01:00
|
|
|
"""
|
2014-05-04 19:31:53 +02:00
|
|
|
:param dict data: volume.partitions part of the manifest
|
2015-01-01 21:09:16 +01:00
|
|
|
:param int sector_size: Sectorsize of the volume
|
2014-05-04 19:31:53 +02:00
|
|
|
:param str bootloader: Name of the bootloader we will use for bootstrapping
|
2014-03-23 16:04:03 +01:00
|
|
|
"""
|
2015-01-01 21:09:16 +01:00
|
|
|
from bootstrapvz.common.sectors import Sectors
|
|
|
|
|
2014-03-23 16:04:03 +01:00
|
|
|
# List of partitions
|
2013-10-05 20:41:05 +02:00
|
|
|
self.partitions = []
|
2013-11-30 14:33:22 +01:00
|
|
|
|
2014-03-23 16:04:03 +01:00
|
|
|
# Returns the last partition unless there is none
|
2013-11-30 14:33:22 +01:00
|
|
|
def last_partition():
|
|
|
|
return self.partitions[-1] if len(self.partitions) > 0 else None
|
2014-01-19 01:02:29 +01:00
|
|
|
|
2015-01-19 01:22:12 +01:00
|
|
|
# The first 34 sectors are reserved for the primary GPT
|
|
|
|
primary_gpt = PartitionGap(Sectors(34, sector_size), last_partition())
|
|
|
|
self.partitions.append(primary_gpt)
|
|
|
|
|
2014-01-19 01:02:29 +01:00
|
|
|
if bootloader == 'grub':
|
2015-01-19 01:22:12 +01:00
|
|
|
# If we are using the grub bootloader we need to create an unformatted partition
|
|
|
|
# at the beginning of the map. Its size is 1007kb, which seems to be chosen so that
|
|
|
|
# gpt_primary + grub = 1024KiB
|
|
|
|
# So lets just specify grub size as 1MiB - 34 sectors
|
2014-01-19 01:02:29 +01:00
|
|
|
from ..partitions.unformatted import UnformattedPartition
|
2015-01-19 01:22:12 +01:00
|
|
|
grub_size = Sectors('1MiB', sector_size) - primary_gpt.size
|
|
|
|
self.grub_boot = UnformattedPartition(grub_size, last_partition())
|
2014-01-19 01:02:29 +01:00
|
|
|
self.partitions.append(self.grub_boot)
|
|
|
|
|
2014-03-23 16:04:03 +01:00
|
|
|
# The boot and swap partitions are optional
|
2013-09-18 00:46:58 +02:00
|
|
|
if 'boot' in data:
|
2015-01-01 21:09:16 +01:00
|
|
|
self.boot = GPTPartition(Sectors(data['boot']['size'], sector_size),
|
2014-02-23 17:52:05 +01:00
|
|
|
data['boot']['filesystem'], data['boot'].get('format_command', None),
|
2014-01-19 12:39:07 +01:00
|
|
|
'boot', last_partition())
|
2015-01-19 01:38:16 +01:00
|
|
|
# Offset all partitions by 1 sector.
|
|
|
|
# parted in jessie has changed and no longer allows partitions to be right next to each other.
|
|
|
|
self.boot.offset = Sectors(1, sector_size)
|
|
|
|
self.boot.size -= self.boot.offset
|
2013-10-05 20:41:05 +02:00
|
|
|
self.partitions.append(self.boot)
|
2013-09-18 00:46:58 +02:00
|
|
|
if 'swap' in data:
|
2015-01-01 21:09:16 +01:00
|
|
|
self.swap = GPTSwapPartition(Sectors(data['swap']['size'], sector_size), last_partition())
|
2015-01-19 01:38:16 +01:00
|
|
|
self.swap.offset = Sectors(1, sector_size)
|
|
|
|
self.swap.size -= self.swap.offset
|
2013-10-05 20:41:05 +02:00
|
|
|
self.partitions.append(self.swap)
|
2015-01-01 21:09:16 +01:00
|
|
|
self.root = GPTPartition(Sectors(data['root']['size'], sector_size),
|
2014-02-23 17:52:05 +01:00
|
|
|
data['root']['filesystem'], data['root'].get('format_command', None),
|
2014-01-25 13:56:28 +01:00
|
|
|
'root', last_partition())
|
2015-01-19 01:38:16 +01:00
|
|
|
self.root.offset = Sectors(1, sector_size)
|
|
|
|
self.root.size -= self.root.offset
|
2013-11-30 14:33:22 +01:00
|
|
|
self.partitions.append(self.root)
|
2013-09-18 00:46:58 +02:00
|
|
|
|
2015-01-19 01:22:12 +01:00
|
|
|
# The last 34 sectors are reserved for the secondary GPT
|
|
|
|
secondary_gpt = PartitionGap(Sectors(34, sector_size), last_partition())
|
|
|
|
self.partitions.append(secondary_gpt)
|
2014-05-04 11:18:54 +02:00
|
|
|
|
2015-01-19 01:22:12 +01:00
|
|
|
# reduce the size of the root partition so that the overall volume size is not exceeded
|
|
|
|
self.root.size -= primary_gpt.size + secondary_gpt.size
|
2014-01-19 12:39:07 +01:00
|
|
|
if hasattr(self, 'grub_boot'):
|
2015-01-19 01:22:12 +01:00
|
|
|
self.root.size -= self.grub_boot.size
|
2015-01-01 21:09:16 +01:00
|
|
|
|
2015-01-19 17:19:27 +01:00
|
|
|
# Set the boot flag on the right partition
|
|
|
|
if hasattr(self, 'grub_boot'):
|
|
|
|
# Mark the partition as a bios_grub partition
|
|
|
|
self.grub_boot.flags.append('bios_grub')
|
|
|
|
else:
|
|
|
|
# Mark the boot partition, or root, if boot does not exist
|
|
|
|
getattr(self, 'boot', self.root).flags.append('legacy_boot')
|
|
|
|
|
2014-01-19 01:02:29 +01:00
|
|
|
super(GPTPartitionMap, self).__init__(bootloader)
|
2013-09-22 21:20:09 +02:00
|
|
|
|
2013-09-25 23:27:31 +02:00
|
|
|
def _before_create(self, event):
|
2014-03-23 16:04:03 +01:00
|
|
|
"""Creates the partition map
|
|
|
|
"""
|
2013-09-25 23:27:31 +02:00
|
|
|
volume = event.volume
|
2014-03-23 16:04:03 +01:00
|
|
|
# Disk alignment still plays a role in virtualized environment,
|
|
|
|
# but I honestly have no clue as to what best practice is here, so we choose 'none'
|
2014-02-23 22:16:10 +01:00
|
|
|
log_check_call(['parted', '--script', '--align', 'none', volume.device_path,
|
2013-09-18 00:46:58 +02:00
|
|
|
'--', 'mklabel', 'gpt'])
|
2014-03-23 16:04:03 +01:00
|
|
|
# Create the partitions
|
2013-09-18 00:46:58 +02:00
|
|
|
for partition in self.partitions:
|
2015-01-01 21:09:16 +01:00
|
|
|
if isinstance(partition, PartitionGap):
|
|
|
|
continue
|
2013-09-18 00:46:58 +02:00
|
|
|
partition.create(volume)
|