Blame SOURCES/0065-libparted-Add-support-for-atari-partition-tables.patch

0cb0b9
From 5750e235a3cb27dc94aae0ca0e7d946c9319cc9d Mon Sep 17 00:00:00 2001
0cb0b9
From: John Paul Adrian Glaubitz <glaubitz@physik.fu-berlin.de>
0cb0b9
Date: Sun, 4 Dec 2016 17:12:46 +0100
0cb0b9
Subject: [PATCH 65/75] libparted: Add support for atari partition tables
0cb0b9
0cb0b9
Signed-off-by: John Paul Adrian Glaubitz <glaubitz@physik.fu-berlin.de>
0cb0b9
Signed-off-by: Brian C. Lane <bcl@redhat.com>
0cb0b9
---
0cb0b9
 libparted/labels/Makefile.am    |    1 +
0cb0b9
 libparted/labels/atari.c        | 1969 +++++++++++++++++++++++++++++++++++++++
0cb0b9
 libparted/labels/pt-limit.gperf |    1 +
0cb0b9
 libparted/libparted.c           |    4 +
0cb0b9
 libparted/tests/common.c        |    3 +
0cb0b9
 po/POTFILES.in                  |    1 +
0cb0b9
 tests/t3310-flags.sh            |    6 +-
0cb0b9
 tests/t9021-maxima.sh           |    5 +-
0cb0b9
 8 files changed, 1988 insertions(+), 2 deletions(-)
0cb0b9
 create mode 100644 libparted/labels/atari.c
0cb0b9
0cb0b9
diff --git a/libparted/labels/Makefile.am b/libparted/labels/Makefile.am
0cb0b9
index c996f81..3327c8c 100644
0cb0b9
--- a/libparted/labels/Makefile.am
0cb0b9
+++ b/libparted/labels/Makefile.am
0cb0b9
@@ -19,6 +19,7 @@ noinst_LTLIBRARIES    =	liblabels.la
0cb0b9
 liblabels_la_SOURCES = \
0cb0b9
   $(S390_SRCS)	\
0cb0b9
   aix.c		\
0cb0b9
+  atari.c	\
0cb0b9
   bsd.c		\
0cb0b9
   dos.c		\
0cb0b9
   dvh.c		\
0cb0b9
diff --git a/libparted/labels/atari.c b/libparted/labels/atari.c
0cb0b9
new file mode 100644
0cb0b9
index 0000000..36e68b9
0cb0b9
--- /dev/null
0cb0b9
+++ b/libparted/labels/atari.c
0cb0b9
@@ -0,0 +1,1969 @@
0cb0b9
+/* -*- Mode: c; tab-width: 8; indent-tabs-mode: nil; c-basic-offset: 8 -*-
0cb0b9
+
0cb0b9
+    libparted - a library for manipulating disk partitions
0cb0b9
+    atari.c - libparted module to manipulate Atari partition tables.
0cb0b9
+    Copyright (C) 2000-2001, 2004, 2007-2014 Free Software Foundation, Inc.
0cb0b9
+
0cb0b9
+    This program is free software; you can redistribute it and/or modify
0cb0b9
+    it under the terms of the GNU General Public License as published by
0cb0b9
+    the Free Software Foundation; either version 3 of the License, or
0cb0b9
+    (at your option) any later version.
0cb0b9
+
0cb0b9
+    This program is distributed in the hope that it will be useful,
0cb0b9
+    but WITHOUT ANY WARRANTY; without even the implied warranty of
0cb0b9
+    MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
0cb0b9
+    GNU General Public License for more details.
0cb0b9
+
0cb0b9
+    You should have received a copy of the GNU General Public License
0cb0b9
+    along with this program.  If not, see <http://www.gnu.org/licenses/>.
0cb0b9
+
0cb0b9
+    Contributor:  Guillaume Knispel <k_guillaume@libertysurf.fr>
0cb0b9
+                  John Paul Adrian Glaubitz <glaubitz@physik.fu-berlin.de>
0cb0b9
+*/
0cb0b9
+
0cb0b9
+/*
0cb0b9
+	Documentation :
0cb0b9
+		README file of atari-fdisk
0cb0b9
+		atari-fdisk source code
0cb0b9
+		Linux atari partitions parser source code
0cb0b9
+			( fs/partitions/atari.[ch] )
0cb0b9
+*/
0cb0b9
+
0cb0b9
+#include <config.h>
0cb0b9
+
0cb0b9
+#include <string.h>
0cb0b9
+#include <parted/parted.h>
0cb0b9
+#include <parted/debug.h>
0cb0b9
+#include <parted/endian.h>
0cb0b9
+#include <string.h>
0cb0b9
+#include <locale.h>
0cb0b9
+#include <xlocale.h>
0cb0b9
+#include <stdint.h>
0cb0b9
+#include <ctype.h>
0cb0b9
+#include <stddef.h>
0cb0b9
+
0cb0b9
+#include "pt-tools.h"
0cb0b9
+
0cb0b9
+#if ENABLE_NLS
0cb0b9
+#  include <libintl.h>
0cb0b9
+#  define _(String) dgettext (PACKAGE, String)
0cb0b9
+#else
0cb0b9
+#  define _(String) (String)
0cb0b9
+#endif /* ENABLE_NLS */
0cb0b9
+
0cb0b9
+
0cb0b9
+/********************** Atari data and structure stuff **********************/
0cb0b9
+
0cb0b9
+#define BOOTABLE_CKSUM		0x1234
0cb0b9
+#define NONBOOT_CKSUM		0x4321
0cb0b9
+
0cb0b9
+#define GEM_MAX			((32*1024*1024)/PED_SECTOR_SIZE_DEFAULT)
0cb0b9
+
0cb0b9
+#define PART_FLAG_USED		0x01
0cb0b9
+#define PART_FLAG_BOOT_GEM	0x80	/* GEMDOS	   */
0cb0b9
+#define PART_FLAG_BOOT_ASV	0x40	/* Atari System V  */
0cb0b9
+#define PART_FLAG_BOOT_BSD	0x20	/* Net(?)BSD	   */
0cb0b9
+#define PART_FLAG_BOOT_LNX	0x10	/* Linux	   */
0cb0b9
+#define PART_FLAG_BOOT_UNK	0x08	/* unknown / other */
0cb0b9
+
0cb0b9
+#define N_AHDI			4
0cb0b9
+#define N_ICD			8
0cb0b9
+
0cb0b9
+#define MAXIMUM_PARTS		64
0cb0b9
+
0cb0b9
+/* what we put instead of id, start and size in empty */
0cb0b9
+/* partition tables, to be able to detect it */
0cb0b9
+#define SIGNATURE_EMPTY_TABLE	"PARTEDATARI"
0cb0b9
+#define SIGNATURE_EMPTY_SIZE	11
0cb0b9
+
0cb0b9
+/* to be compared to the last two bytes of 1st sector (Big Endian) */
0cb0b9
+static const uint16_t atr_forbidden_sign[] = {
0cb0b9
+	0x55AA,
0cb0b9
+	0
0cb0b9
+};
0cb0b9
+
0cb0b9
+static const char *atr_known_icd_pid[] = {
0cb0b9
+	"BGM", "GEM", "LNX", "SWP", "RAW", NULL
0cb0b9
+};
0cb0b9
+
0cb0b9
+/* static const char *atr_known_pid[] = { */
0cb0b9
+/* 	"BGM", "GEM", "LNX", "MAC", "MIX", "MNX", "RAW", "SWP", "UNX", */
0cb0b9
+/* 	"F32", "SV4", NULL */
0cb0b9
+/* }; */
0cb0b9
+
0cb0b9
+struct _AtariPartID2BootFlag {
0cb0b9
+	const char	pid[4];
0cb0b9
+	uint8_t		flag;
0cb0b9
+};
0cb0b9
+typedef struct _AtariPartID2BootFlag AtariPartID2BootFlag;
0cb0b9
+
0cb0b9
+static AtariPartID2BootFlag atr_pid2bf[] = {
0cb0b9
+	{ "GEM", PART_FLAG_BOOT_GEM },
0cb0b9
+	{ "BGM", PART_FLAG_BOOT_GEM },
0cb0b9
+	{ "UNX", PART_FLAG_BOOT_ASV },
0cb0b9
+	{ "LNX", PART_FLAG_BOOT_LNX },
0cb0b9
+	{ "",    PART_FLAG_BOOT_UNK },
0cb0b9
+};
0cb0b9
+
0cb0b9
+struct _AtariFS2PartId {
0cb0b9
+	const char*	fs;
0cb0b9
+	const char	pid[4];
0cb0b9
+	PedSector	max_sectors;
0cb0b9
+};
0cb0b9
+typedef struct _AtariFS2PartId AtariFS2PartId;
0cb0b9
+
0cb0b9
+static AtariFS2PartId atr_fs2pid[] = {
0cb0b9
+/* Other ID are available : MIX MNX <= minix
0cb0b9
+				UNX <= Atari SysV Unix
0cb0b9
+				SV4 <= Univ System 4   */
0cb0b9
+	{ "ext2",	"LNX", INT32_MAX },
0cb0b9
+	{ "ext3",	"LNX", INT32_MAX },
0cb0b9
+	{ "fat16",	"GEM",   GEM_MAX },	/* small partitions */
0cb0b9
+	{ "fat16",	"BGM", INT32_MAX },	/* big partitions */
0cb0b9
+	{ "fat32",	"F32", INT32_MAX },
0cb0b9
+	{ "hfs",	"MAC", INT32_MAX },
0cb0b9
+	{ "hfs+",	"MAC", INT32_MAX },
0cb0b9
+	{ "hfsx",	"MAC", INT32_MAX },
0cb0b9
+	{ "jfs",	"LNX", INT32_MAX },
0cb0b9
+	{ "linux-swap", "SWP", INT32_MAX },
0cb0b9
+	{ "reiserfs",	"LNX", INT32_MAX },
0cb0b9
+	{ "hp-ufs",	"LNX", INT32_MAX },
0cb0b9
+	{ "sun-ufs",	"LNX", INT32_MAX },
0cb0b9
+	{ "xfs",	"LNX", INT32_MAX },
0cb0b9
+	{ "ntfs",	"RAW", INT32_MAX },
0cb0b9
+	{ "",		"RAW", INT32_MAX },	/* default entry */
0cb0b9
+	{ NULL,		 ""  ,         0 }	/* end of list */
0cb0b9
+};
0cb0b9
+
0cb0b9
+struct __attribute__ ((packed)) _AtariRawPartition {
0cb0b9
+	uint8_t		flag;	/* bit 0: active; bit 7: bootable */
0cb0b9
+	uint8_t		id[3];	/* "GEM", "BGM", "XGM", ... */
0cb0b9
+	uint32_t	start;	/* start of partition */
0cb0b9
+	uint32_t	size;	/* length of partition */
0cb0b9
+};
0cb0b9
+typedef struct _AtariRawPartition AtariRawPartition;
0cb0b9
+
0cb0b9
+struct __attribute__ ((packed)) _AtariRawTable {
0cb0b9
+	uint8_t		  boot_code[0x156]; /* room for boot code */
0cb0b9
+	AtariRawPartition icd_part[N_ICD];  /* info for ICD-partitions 5..12 */
0cb0b9
+	uint8_t		  unused[0xc];
0cb0b9
+	uint32_t	  hd_size;	/* size of disk in blocks */
0cb0b9
+	AtariRawPartition part[N_AHDI];	/* the four primary partitions */
0cb0b9
+	uint32_t	  bsl_start;	/* start of bad sector list */
0cb0b9
+	uint32_t	  bsl_count;	/* length of bad sector list */
0cb0b9
+	uint16_t	  checksum;	/* checksum for bootable disks */
0cb0b9
+};
0cb0b9
+typedef struct _AtariRawTable AtariRawTable;
0cb0b9
+
0cb0b9
+typedef enum {
0cb0b9
+	FMT_AHDI = 0,	/* AHDI v1 compatible, no ICD and no XGM */
0cb0b9
+	FMT_XGM  = 1,	/* AHDI v3 with XGM / this disable ICD */
0cb0b9
+	FMT_ICD  = 2	/* ICD detected / requested because more than 4 prim */
0cb0b9
+			/* no XGM allowed */
0cb0b9
+} AtrFmt;
0cb0b9
+
0cb0b9
+struct _AtariDisk {
0cb0b9
+	AtrFmt	format;
0cb0b9
+	int	has_been_read;	/* actually means has been read or written... */
0cb0b9
+	uint32_t bsl_start;	/* first sector of the Bad Sectors List */
0cb0b9
+	uint32_t bsl_count;	/* number of sectors of the BSL */
0cb0b9
+	uint8_t	HDX_comp;	/* if set to one, atari_write will initialize */
0cb0b9
+				/* the bsl area */
0cb0b9
+};
0cb0b9
+typedef struct _AtariDisk AtariDisk;
0cb0b9
+
0cb0b9
+struct _AtariPart {
0cb0b9
+	char	part_id[4];	/* ASCIIZ */
0cb0b9
+	char	icd_id[4];	/* Linux only parse a limited set of ID */
0cb0b9
+				/* in ICD (why???), so everything else  */
0cb0b9
+				/* is translated to RAW.		*/
0cb0b9
+	uint8_t	flag;		/* without bit 0 (entry used) */
0cb0b9
+};
0cb0b9
+typedef struct _AtariPart AtariPart;
0cb0b9
+
0cb0b9
+/* set by initialisation code to C locale */
0cb0b9
+static locale_t atr_c_locale;
0cb0b9
+
0cb0b9
+static PedDiskType atari_disk_type;
0cb0b9
+
0cb0b9
+
0cb0b9
+
0cb0b9
+/******************************** Atari Code ********************************/
0cb0b9
+
0cb0b9
+#define ATARI_DISK(disk)	((AtariDisk*)((disk)->disk_specific))
0cb0b9
+#define ATARI_PART(part)	((AtariPart*)((part)->disk_specific))
0cb0b9
+
0cb0b9
+#define atr_pid_eq(a,b)		(!memcmp( (a), (b), 3 ))
0cb0b9
+
0cb0b9
+#define atr_pid_assign(a, b)	(memcpy ( (a), (b), 3 ))
0cb0b9
+
0cb0b9
+#define atr_part_used(part)	(((part)->flag) & PART_FLAG_USED)
0cb0b9
+
0cb0b9
+static int
0cb0b9
+atr_start_size_correct (uint32_t start, uint32_t size, uint32_t hd_size)
0cb0b9
+{
0cb0b9
+	uint32_t end = start + size;
0cb0b9
+
0cb0b9
+	return  end >= start
0cb0b9
+		&& 0 < start && start <= hd_size
0cb0b9
+		&& 0 < size && size <= hd_size
0cb0b9
+		&& 0 < end && end <= hd_size;
0cb0b9
+}
0cb0b9
+
0cb0b9
+static int
0cb0b9
+atr_part_correct (AtariRawPartition* part, uint32_t hd_size)
0cb0b9
+{
0cb0b9
+	uint32_t start, size;
0cb0b9
+
0cb0b9
+	start = PED_BE32_TO_CPU (part->start);
0cb0b9
+	size = PED_BE32_TO_CPU (part->size);
0cb0b9
+
0cb0b9
+	return isalnum_l(part->id[0], atr_c_locale)
0cb0b9
+		&& isalnum_l(part->id[1], atr_c_locale)
0cb0b9
+		&& isalnum_l(part->id[2], atr_c_locale)
0cb0b9
+		&& atr_start_size_correct (start, size, hd_size);
0cb0b9
+}
0cb0b9
+
0cb0b9
+static int _GL_ATTRIBUTE_PURE
0cb0b9
+atr_pid_known (const char* pid, const char** pid_list)
0cb0b9
+{
0cb0b9
+	for (; *pid_list; pid_list++) {
0cb0b9
+		if (atr_pid_eq(pid, *pid_list))
0cb0b9
+			return 1;
0cb0b9
+	}
0cb0b9
+
0cb0b9
+	return 0;
0cb0b9
+}
0cb0b9
+
0cb0b9
+/* Recognize Parted signature in an AHDI entry, used to
0cb0b9
+ * identify empty Atari partition tables */
0cb0b9
+static int
0cb0b9
+atr_is_signature_entry (AtariRawPartition* part)
0cb0b9
+{
0cb0b9
+	return part->flag == 0
0cb0b9
+		&& !memcmp (part->id, SIGNATURE_EMPTY_TABLE,
0cb0b9
+				      SIGNATURE_EMPTY_SIZE );
0cb0b9
+}
0cb0b9
+
0cb0b9
+/* Set Parted signature in an AHDI entry */
0cb0b9
+static void
0cb0b9
+atr_put_signature_entry (AtariRawPartition* part)
0cb0b9
+{
0cb0b9
+	part->flag = 0;
0cb0b9
+	memcpy (part->id, SIGNATURE_EMPTY_TABLE, SIGNATURE_EMPTY_SIZE);
0cb0b9
+}
0cb0b9
+
0cb0b9
+#define atr_part_known(part, pid_list) (atr_pid_known ((part)->id, pid_list))
0cb0b9
+
0cb0b9
+#define atr_part_valid(part, sz) (atr_part_used(part)\
0cb0b9
+				  && atr_part_correct((part), (sz)))
0cb0b9
+#define atr_part_trash(part, sz) (atr_part_used(part)\
0cb0b9
+				  && !atr_part_correct((part), (sz)))
0cb0b9
+
0cb0b9
+/* Check if this device can be used with an Atari label */
0cb0b9
+static int
0cb0b9
+atr_can_use_dev (const PedDevice *dev)
0cb0b9
+{
0cb0b9
+	/* i really don't know how atari behave with non 512 bytes */
0cb0b9
+	/* sectors... */
0cb0b9
+	if (dev->sector_size != PED_SECTOR_SIZE_DEFAULT) {
0cb0b9
+		ped_exception_throw (
0cb0b9
+			PED_EXCEPTION_ERROR,
0cb0b9
+			PED_EXCEPTION_CANCEL,
0cb0b9
+			_("Can't use Atari partition tables on disks with a "
0cb0b9
+			  "sector size not equal to %d bytes."),
0cb0b9
+			(int)PED_SECTOR_SIZE_DEFAULT );
0cb0b9
+		return 0;
0cb0b9
+	}
0cb0b9
+
0cb0b9
+	/* the format isn't well defined enough to support > 0x7FFFFFFF */
0cb0b9
+	/* sectors */
0cb0b9
+	if (dev->length > INT32_MAX) {
0cb0b9
+		ped_exception_throw (
0cb0b9
+			PED_EXCEPTION_ERROR,
0cb0b9
+			PED_EXCEPTION_CANCEL,
0cb0b9
+			_("Can't use Atari partition tables on disks with more "
0cb0b9
+			  "than %d sectors."),
0cb0b9
+			INT32_MAX );
0cb0b9
+		return 0;
0cb0b9
+	}
0cb0b9
+
0cb0b9
+	return 1;
0cb0b9
+}
0cb0b9
+
0cb0b9
+/*
0cb0b9
+ * The Atari disk label doesn't have any magic id
0cb0b9
+ * so we must completely parse the layout to be sure
0cb0b9
+ * we are really dealing with it.
0cb0b9
+ */
0cb0b9
+static int
0cb0b9
+atari_probe (const PedDevice *dev)
0cb0b9
+{
0cb0b9
+	AtariRawTable	table;
0cb0b9
+	uint32_t	rs_hd_size, parts, exts;
0cb0b9
+	int		valid_count, xgm_part, xgm_num, i;
0cb0b9
+	int		num_sign, total_count = 0;
0cb0b9
+
0cb0b9
+	PED_ASSERT (dev != NULL);
0cb0b9
+
0cb0b9
+	/* Device Spec ok for Atari label? */
0cb0b9
+	if (!atr_can_use_dev (dev))
0cb0b9
+		return 0;
0cb0b9
+
0cb0b9
+	/* read the root sector */
0cb0b9
+	if (!ped_device_read (dev, &table, 0, 1))
0cb0b9
+		return 0;
0cb0b9
+
0cb0b9
+	/* number of sectors stored in the root sector > device length ? */
0cb0b9
+	/* => just reject the Atari disk label */
0cb0b9
+	rs_hd_size = PED_BE32_TO_CPU (table.hd_size);
0cb0b9
+	if (rs_hd_size > dev->length
0cb0b9
+	    || rs_hd_size < 2)
0cb0b9
+		return 0;
0cb0b9
+
0cb0b9
+	/* check the BSL fields */
0cb0b9
+	if ((table.bsl_start || table.bsl_count)
0cb0b9
+	    && !atr_start_size_correct (PED_BE32_TO_CPU (table.bsl_start),
0cb0b9
+					PED_BE32_TO_CPU (table.bsl_count),
0cb0b9
+					rs_hd_size ) )
0cb0b9
+		return 0;
0cb0b9
+
0cb0b9
+	/* scan the main AHDI fields */
0cb0b9
+	num_sign = 0; xgm_num = 0;
0cb0b9
+	valid_count = 0; xgm_part = 0;
0cb0b9
+	for (i = 0; i < N_AHDI; i++) {
0cb0b9
+		if (atr_part_valid (&table.part[i], rs_hd_size)) {
0cb0b9
+			valid_count++;
0cb0b9
+			total_count++;
0cb0b9
+			if (atr_pid_eq(table.part[i].id, "XGM")) {
0cb0b9
+				xgm_part++;
0cb0b9
+				xgm_num = i;
0cb0b9
+			}
0cb0b9
+		} else if (atr_part_trash (&table.part[i], rs_hd_size)) {
0cb0b9
+			return 0;
0cb0b9
+		}
0cb0b9
+		if (atr_is_signature_entry (&table.part[i]))
0cb0b9
+			num_sign++;
0cb0b9
+	}
0cb0b9
+
0cb0b9
+	/* no way to reliably detect empty Atari disk labels if
0cb0b9
+	 *    they aren't using parted signature in 4 prim fields
0cb0b9
+	 * && reject multi XGM labels because Parted can't handle
0cb0b9
+	 *    multiple extended partitions
0cb0b9
+	 * && reject if xgm partition in slot 0 because not allowed */
0cb0b9
+	if ((!valid_count && num_sign != N_AHDI)
0cb0b9
+	    || xgm_part > 1
0cb0b9
+	    || (xgm_part == 1 && xgm_num == 0) )
0cb0b9
+		return 0;
0cb0b9
+
0cb0b9
+	/* check coherency of each logical partitions and ARS */
0cb0b9
+	if (xgm_part) { /* ! WARNING ! reuses "table" */
0cb0b9
+		/* we must allow empty ext partition even if they're   */
0cb0b9
+		/* not valid because parted write the layout to the HD */
0cb0b9
+		/* at each operation, and we can't create ext and log  */
0cb0b9
+		/* at the same time */
0cb0b9
+		int	empty_ars_allowed = 1;
0cb0b9
+
0cb0b9
+		parts = exts = PED_BE32_TO_CPU (table.part[xgm_num].start);
0cb0b9
+		while (1) {
0cb0b9
+			if (!ped_device_read (dev, &table, parts, 1))
0cb0b9
+				return 0;
0cb0b9
+
0cb0b9
+			for (i = 0; i < N_AHDI-1; ++i) {
0cb0b9
+				if (atr_part_used (&table.part[i]))
0cb0b9
+					break;
0cb0b9
+			}
0cb0b9
+
0cb0b9
+			/* we allow the ext part to be empty (see above) */
0cb0b9
+			if (i == N_AHDI-1 && empty_ars_allowed)
0cb0b9
+				break;
0cb0b9
+
0cb0b9
+			/* data partition must be in slot 0, 1 or 2 */
0cb0b9
+			if (i == N_AHDI-1
0cb0b9
+			    || !atr_part_correct (&table.part[i], rs_hd_size
0cb0b9
+								  - parts )
0cb0b9
+			    || atr_pid_eq (table.part[i].id, "XGM"))
0cb0b9
+				return 0;
0cb0b9
+
0cb0b9
+			/* If there is at least one logical partition */
0cb0b9
+			/* then next ARS should not be empty */
0cb0b9
+			empty_ars_allowed = 0;
0cb0b9
+
0cb0b9
+			total_count++;
0cb0b9
+			if (total_count > MAXIMUM_PARTS) {
0cb0b9
+				ped_exception_throw (
0cb0b9
+					PED_EXCEPTION_ERROR,
0cb0b9
+					PED_EXCEPTION_CANCEL,
0cb0b9
+					_("Too many Atari partitions detected. "
0cb0b9
+					" Maybe there is a loop in the XGM "
0cb0b9
+					"linked list.  Aborting.") );
0cb0b9
+				return 0;
0cb0b9
+			}
0cb0b9
+
0cb0b9
+			/* end of logical partitions? */
0cb0b9
+			if (!atr_part_used (&table.part[i+1]))
0cb0b9
+				break;
0cb0b9
+
0cb0b9
+			/* is this really the descriptor of the next ARS? */
0cb0b9
+			if (!atr_part_correct (&table.part[i+1], rs_hd_size
0cb0b9
+								  - exts )
0cb0b9
+			    || !atr_pid_eq (table.part[i+1].id, "XGM"))
0cb0b9
+				return 0;
0cb0b9
+
0cb0b9
+			parts = exts + PED_BE32_TO_CPU (table.part[i+1].start);
0cb0b9
+		}
0cb0b9
+	} /* no XGM so try ICD */
0cb0b9
+	  else if (atr_part_valid (&table.icd_part[0], rs_hd_size)
0cb0b9
+		   && atr_part_known (&table.icd_part[0], atr_known_icd_pid)) {
0cb0b9
+		for (i = 1; i < N_ICD; i++) {
0cb0b9
+			if (atr_part_trash (&table.icd_part[i], rs_hd_size))
0cb0b9
+				return 0;
0cb0b9
+		}
0cb0b9
+	}
0cb0b9
+
0cb0b9
+	return 1;
0cb0b9
+}
0cb0b9
+
0cb0b9
+static void
0cb0b9
+atr_disk_reset (AtariDisk* atr_disk)
0cb0b9
+{
0cb0b9
+	/* Empty partition table => only AHDI needed right now */
0cb0b9
+	atr_disk->format = FMT_AHDI;
0cb0b9
+	/* The disk is not in sync with the actual content of the label */
0cb0b9
+	atr_disk->has_been_read = 0;
0cb0b9
+	/* Create an empty BSL for HDX compatibility */
0cb0b9
+	atr_disk->bsl_start = 1;
0cb0b9
+	atr_disk->bsl_count = 1;
0cb0b9
+	atr_disk->HDX_comp = 1;
0cb0b9
+}
0cb0b9
+
0cb0b9
+/*
0cb0b9
+ * Must set up the PedDisk and the associated AtariDisk as if
0cb0b9
+ * the user is doing mklabel, since in this case atari_alloc
0cb0b9
+ * is called alone whereas when reading an existing partition
0cb0b9
+ * table atari_read is called after atari_alloc and can overwrite
0cb0b9
+ * the settings.
0cb0b9
+ */
0cb0b9
+static PedDisk*
0cb0b9
+atari_alloc (const PedDevice* dev)
0cb0b9
+{
0cb0b9
+	PedDisk*	disk;
0cb0b9
+	AtariDisk*	atr_disk;
0cb0b9
+
0cb0b9
+	PED_ASSERT (dev != NULL);
0cb0b9
+
0cb0b9
+	if (!atr_can_use_dev (dev)
0cb0b9
+	    || !(disk = _ped_disk_alloc (dev, &atari_disk_type)))
0cb0b9
+		return NULL;
0cb0b9
+
0cb0b9
+	if (!(disk->disk_specific = atr_disk = ped_malloc (sizeof (AtariDisk))))
0cb0b9
+		goto error_free_disk;
0cb0b9
+
0cb0b9
+	atr_disk_reset (atr_disk);
0cb0b9
+
0cb0b9
+	return disk;
0cb0b9
+
0cb0b9
+error_free_disk:
0cb0b9
+	free (disk);
0cb0b9
+	return NULL;
0cb0b9
+}
0cb0b9
+
0cb0b9
+static PedDisk*
0cb0b9
+atari_duplicate (const PedDisk* disk)
0cb0b9
+{
0cb0b9
+	PedDisk*	new_disk;
0cb0b9
+	AtariDisk*	old_atr_dsk;
0cb0b9
+	AtariDisk*	new_atr_dsk;
0cb0b9
+
0cb0b9
+	PED_ASSERT (disk != NULL);
0cb0b9
+	PED_ASSERT (disk->dev != NULL);
0cb0b9
+	PED_ASSERT (disk->disk_specific != NULL);
0cb0b9
+
0cb0b9
+	old_atr_dsk = ATARI_DISK (disk);
0cb0b9
+	if (!(new_disk = ped_disk_new_fresh (disk->dev, &atari_disk_type)))
0cb0b9
+		return NULL;
0cb0b9
+	new_atr_dsk = ATARI_DISK (new_disk);
0cb0b9
+
0cb0b9
+	memcpy (new_atr_dsk, old_atr_dsk, sizeof(*old_atr_dsk));
0cb0b9
+
0cb0b9
+	return new_disk;
0cb0b9
+}
0cb0b9
+
0cb0b9
+static void
0cb0b9
+atari_free (PedDisk* disk)
0cb0b9
+{
0cb0b9
+	AtariDisk* atr_disk;
0cb0b9
+	PED_ASSERT (disk != NULL);
0cb0b9
+	PED_ASSERT (disk->disk_specific != NULL);
0cb0b9
+	atr_disk = ATARI_DISK (disk);
0cb0b9
+
0cb0b9
+	_ped_disk_free (disk);
0cb0b9
+	free (atr_disk);
0cb0b9
+}
0cb0b9
+
0cb0b9
+/* Warning : ID not ASCIIZ but 3 chars long */
0cb0b9
+static void
0cb0b9
+atr_part_sysraw (PedPartition* part, const char* id, uint8_t flag)
0cb0b9
+{
0cb0b9
+	AtariPart* atr_part = ATARI_PART (part);
0cb0b9
+
0cb0b9
+	atr_part->flag = flag & ~PART_FLAG_USED;
0cb0b9
+
0cb0b9
+	atr_pid_assign (atr_part->part_id, id);
0cb0b9
+	atr_part->part_id[3] = 0;
0cb0b9
+
0cb0b9
+	if (atr_pid_known (id, atr_known_icd_pid)) {
0cb0b9
+		atr_pid_assign (atr_part->icd_id, id);
0cb0b9
+		atr_part->icd_id[3] = 0;
0cb0b9
+	} else {
0cb0b9
+		atr_pid_assign (atr_part->icd_id, "RAW");
0cb0b9
+		atr_part->icd_id[3] = 0;
0cb0b9
+	}
0cb0b9
+}
0cb0b9
+
0cb0b9
+static int
0cb0b9
+atr_parse_add_rawpart (PedDisk* disk, PedPartitionType type, PedSector st_off,
0cb0b9
+		       int num, const AtariRawPartition* rawpart )
0cb0b9
+{
0cb0b9
+	PedSector	start, end;
0cb0b9
+	PedPartition* 	part;
0cb0b9
+	PedConstraint*	const_exact;
0cb0b9
+	int		added;
0cb0b9
+
0cb0b9
+	start = st_off + PED_BE32_TO_CPU (rawpart->start);
0cb0b9
+	end = start + PED_BE32_TO_CPU (rawpart->size) - 1;
0cb0b9
+
0cb0b9
+	part = ped_partition_new (disk, type, NULL, start, end);
0cb0b9
+	if (!part)
0cb0b9
+		return 0;
0cb0b9
+
0cb0b9
+	/*part->num = num;*/	/* Enumeration will take care of that */
0cb0b9
+	part->num = -1;		/* Indeed we can't enumerate here
0cb0b9
+				 * because the enumerate function uses
0cb0b9
+				 * -1 do detect new partition being
0cb0b9
+				 * inserted and update the atrdisk->format */
0cb0b9
+	if (type != PED_PARTITION_EXTENDED)
0cb0b9
+		part->fs_type = ped_file_system_probe (&part->geom);
0cb0b9
+	else
0cb0b9
+		part->fs_type = NULL;
0cb0b9
+	atr_part_sysraw (part, rawpart->id, rawpart->flag);
0cb0b9
+
0cb0b9
+	const_exact = ped_constraint_exact (&part->geom);
0cb0b9
+	added = ped_disk_add_partition (disk, part, const_exact);
0cb0b9
+	ped_constraint_destroy (const_exact);
0cb0b9
+	if (!added) {
0cb0b9
+		ped_partition_destroy (part);
0cb0b9
+		return 0;
0cb0b9
+	}
0cb0b9
+
0cb0b9
+	PED_ASSERT (part->num == num);
0cb0b9
+	return 1;
0cb0b9
+}
0cb0b9
+
0cb0b9
+/*
0cb0b9
+ * Read the chained list of logical partitions.
0cb0b9
+ * exts points to the first Auxiliary Root Sector, at the start
0cb0b9
+ * of the extended partition.
0cb0b9
+ * In each ARS one partition entry describes to the logical partition
0cb0b9
+ * (start relative to the ARS position) and the next entry with ID "XGM"
0cb0b9
+ * points to the next ARS (start relative to exts).
0cb0b9
+ */
0cb0b9
+static int
0cb0b9
+atr_read_logicals (PedDisk* disk, PedSector exts, int* pnum)
0cb0b9
+{
0cb0b9
+	AtariRawTable	table;
0cb0b9
+	PedSector	parts = exts;
0cb0b9
+	int		i, empty_ars_allowed = 1;
0cb0b9
+
0cb0b9
+	while (1) {
0cb0b9
+		if (!ped_device_read (disk->dev, &table, parts, 1))
0cb0b9
+			return 0;
0cb0b9
+
0cb0b9
+		for (i = 0; i < N_AHDI-1; ++i)
0cb0b9
+			if (atr_part_used (&table.part[i]))
0cb0b9
+				break;
0cb0b9
+
0cb0b9
+		if (i == N_AHDI-1 && empty_ars_allowed)
0cb0b9
+			break;
0cb0b9
+
0cb0b9
+		/* data partition must be in slot 0, 1 or 2 */
0cb0b9
+		if (i == N_AHDI-1
0cb0b9
+		    || atr_pid_eq (table.part[i].id, "XGM")) {
0cb0b9
+			ped_exception_throw (
0cb0b9
+				PED_EXCEPTION_ERROR,
0cb0b9
+				PED_EXCEPTION_CANCEL,
0cb0b9
+				_("No data partition found in the ARS at "
0cb0b9
+				  "sector %lli."), parts );
0cb0b9
+			return 0;
0cb0b9
+		}
0cb0b9
+
0cb0b9
+		empty_ars_allowed = 0;
0cb0b9
+
0cb0b9
+		if (!atr_parse_add_rawpart (disk, PED_PARTITION_LOGICAL,
0cb0b9
+					    parts, *pnum, &table.part[i] ) )
0cb0b9
+			return 0;
0cb0b9
+
0cb0b9
+		(*pnum)++;
0cb0b9
+
0cb0b9
+		/* end of logical partitions? */
0cb0b9
+		if (!atr_part_used (&table.part[i+1]))
0cb0b9
+			break;
0cb0b9
+
0cb0b9
+		if (!atr_pid_eq (table.part[i+1].id, "XGM")) {
0cb0b9
+			ped_exception_throw (
0cb0b9
+				PED_EXCEPTION_ERROR,
0cb0b9
+				PED_EXCEPTION_CANCEL,
0cb0b9
+				_("The entry of the next logical ARS is not of "
0cb0b9
+				  "type XGM in ARS at sector %lli."), parts );
0cb0b9
+			return 0;
0cb0b9
+		}
0cb0b9
+
0cb0b9
+		parts = exts + PED_BE32_TO_CPU (table.part[i+1].start);
0cb0b9
+	}
0cb0b9
+
0cb0b9
+	return 1;
0cb0b9
+}
0cb0b9
+
0cb0b9
+static int
0cb0b9
+atari_read (PedDisk* disk)
0cb0b9
+{
0cb0b9
+	AtariRawTable	table;
0cb0b9
+	AtariDisk*	atr_disk;
0cb0b9
+	uint32_t	rs_hd_size;
0cb0b9
+	int		i, pnum, xgm, pcount;
0cb0b9
+
0cb0b9
+	PED_ASSERT (disk != NULL);
0cb0b9
+	PED_ASSERT (disk->dev != NULL);
0cb0b9
+	PED_ASSERT (disk->disk_specific != NULL);
0cb0b9
+	atr_disk = ATARI_DISK (disk);
0cb0b9
+
0cb0b9
+	ped_disk_delete_all (disk);
0cb0b9
+	atr_disk_reset (atr_disk);
0cb0b9
+
0cb0b9
+	if (!atari_probe (disk->dev)) {
0cb0b9
+		if (ped_exception_throw (
0cb0b9
+			PED_EXCEPTION_ERROR,
0cb0b9
+			PED_EXCEPTION_IGNORE_CANCEL,
0cb0b9
+			_("There doesn't seem to be an Atari partition table "
0cb0b9
+			  "on this disk (%s), or it is corrupted."),
0cb0b9
+			disk->dev->path )
0cb0b9
+				!= PED_EXCEPTION_IGNORE)
0cb0b9
+			return 0;
0cb0b9
+	}
0cb0b9
+
0cb0b9
+	if (!ped_device_read (disk->dev, (void*) &table, 0, 1))
0cb0b9
+		goto error;
0cb0b9
+
0cb0b9
+	/* We are sure that the layout looks coherent so we
0cb0b9
+	   don't need to check too much */
0cb0b9
+
0cb0b9
+	rs_hd_size = PED_BE32_TO_CPU (table.hd_size);
0cb0b9
+	atr_disk->bsl_start = PED_BE32_TO_CPU (table.bsl_start);
0cb0b9
+	atr_disk->bsl_count = PED_BE32_TO_CPU (table.bsl_count);
0cb0b9
+	atr_disk->HDX_comp = 0;
0cb0b9
+
0cb0b9
+	/* AHDI primary partitions */
0cb0b9
+	pnum = 1; xgm = 0; pcount = 0;
0cb0b9
+	for (i = 0; i < N_AHDI; i++) {
0cb0b9
+		if (!atr_part_used (&table.part[i]))
0cb0b9
+			continue;
0cb0b9
+
0cb0b9
+		pcount++;
0cb0b9
+
0cb0b9
+		if (atr_pid_eq (table.part[i].id, "XGM")) {
0cb0b9
+
0cb0b9
+			atr_disk->format = FMT_XGM;
0cb0b9
+			xgm = 1;
0cb0b9
+			if (!atr_parse_add_rawpart(disk, PED_PARTITION_EXTENDED,
0cb0b9
+						   0, 0, &table.part[i] )
0cb0b9
+			    || !atr_read_logicals (
0cb0b9
+					disk,
0cb0b9
+					PED_BE32_TO_CPU (table.part[i].start),
0cb0b9
+					&pnum ) )
0cb0b9
+				goto error;
0cb0b9
+
0cb0b9
+		} else {
0cb0b9
+
0cb0b9
+			if (!atr_parse_add_rawpart (disk, PED_PARTITION_NORMAL,
0cb0b9
+						    0, pnum, &table.part[i] ) )
0cb0b9
+				goto error;
0cb0b9
+			pnum++;
0cb0b9
+		}
0cb0b9
+	}
0cb0b9
+
0cb0b9
+	/* If no XGM partition has been found, the AHDI table is not empty,  */
0cb0b9
+	/* the first entry is valid and its ID ok for ICD, then we parse the */
0cb0b9
+	/* ICD table. */
0cb0b9
+	if (!xgm && pcount != 0
0cb0b9
+		 && atr_part_valid (&table.icd_part[0], rs_hd_size)
0cb0b9
+		 && atr_part_known (&table.icd_part[0], atr_known_icd_pid))
0cb0b9
+	for (i = 0; i < N_ICD; i++) {
0cb0b9
+
0cb0b9
+		if (!atr_part_known (&table.icd_part[i], atr_known_icd_pid)
0cb0b9
+		    || !atr_part_used (&table.icd_part[i]))
0cb0b9
+			continue;
0cb0b9
+		atr_disk->format = FMT_ICD;
0cb0b9
+
0cb0b9
+		if (!atr_parse_add_rawpart (disk, PED_PARTITION_NORMAL,
0cb0b9
+					    0, pnum, &table.icd_part[i] ) )
0cb0b9
+			goto error;
0cb0b9
+		pnum++;
0cb0b9
+	}
0cb0b9
+
0cb0b9
+	atr_disk->has_been_read = 1;
0cb0b9
+	return 1;
0cb0b9
+
0cb0b9
+error:
0cb0b9
+	ped_disk_delete_all (disk);
0cb0b9
+	atr_disk_reset (atr_disk);
0cb0b9
+	return 0;
0cb0b9
+}
0cb0b9
+
0cb0b9
+/* Returns the number of the first logical partition or -1 if not found */
0cb0b9
+static int
0cb0b9
+atr_find_first_log (const PedDisk* disk)
0cb0b9
+{
0cb0b9
+	PedPartition*	part;
0cb0b9
+	int		first_log, last;
0cb0b9
+
0cb0b9
+	last = ped_disk_get_last_partition_num (disk);
0cb0b9
+
0cb0b9
+	for (first_log = 1; first_log <= last; first_log++) {
0cb0b9
+		if ((part = ped_disk_get_partition (disk, first_log))
0cb0b9
+		     && (part->type & PED_PARTITION_LOGICAL))
0cb0b9
+			break;
0cb0b9
+	}
0cb0b9
+
0cb0b9
+	return first_log > last ? -1 : first_log;
0cb0b9
+}
0cb0b9
+
0cb0b9
+#ifndef DISCOVER_ONLY
0cb0b9
+static int
0cb0b9
+atari_clobber (PedDevice* dev)
0cb0b9
+{
0cb0b9
+	AtariRawTable table;
0cb0b9
+
0cb0b9
+	PED_ASSERT (dev != NULL);
0cb0b9
+	PED_ASSERT (atari_probe (dev));
0cb0b9
+
0cb0b9
+	if (!ped_device_read (dev, &table, 0, 1))
0cb0b9
+		return 0;
0cb0b9
+
0cb0b9
+	/* clear anything but the boot code and the optional ICD table */
0cb0b9
+	memset (table.boot_code + offsetof (AtariRawTable, hd_size),
0cb0b9
+		0,
0cb0b9
+		PED_SECTOR_SIZE_DEFAULT - offsetof (AtariRawTable, hd_size));
0cb0b9
+
0cb0b9
+	return ped_device_write (dev, &table, 0, 1);
0cb0b9
+}
0cb0b9
+
0cb0b9
+/* Computes the checksum of the root sector */
0cb0b9
+static uint16_t
0cb0b9
+atr_calc_rs_sum (const AtariRawTable* table)
0cb0b9
+{
0cb0b9
+	const uint16_t* word = (uint16_t*)(table);
0cb0b9
+	const uint16_t* end  = (uint16_t*)(table + 1);
0cb0b9
+	uint16_t sum;
0cb0b9
+
0cb0b9
+	for (sum = 0; word < end; word++)
0cb0b9
+		sum += PED_BE16_TO_CPU(*word);
0cb0b9
+
0cb0b9
+	return sum;
0cb0b9
+}
0cb0b9
+
0cb0b9
+/* Returns 1 if the root sector is bootable, else returns 0 */
0cb0b9
+static int
0cb0b9
+atr_is_boot_table (const AtariRawTable* table)
0cb0b9
+{
0cb0b9
+	return atr_calc_rs_sum (table) == BOOTABLE_CKSUM;
0cb0b9
+}
0cb0b9
+
0cb0b9
+/*
0cb0b9
+ * Returns 1 if sign belongs to a set of `forbidden' signatures.
0cb0b9
+ * (e.g.: 55AA which is the MSDOS siganture...)
0cb0b9
+ * Only used for non bootable root sector since the signature of
0cb0b9
+ * a bootable one is unique.
0cb0b9
+ */
0cb0b9
+static int _GL_ATTRIBUTE_PURE
0cb0b9
+atr_sign_is_forbidden (uint16_t sign)
0cb0b9
+{
0cb0b9
+	const uint16_t* forbidden;
0cb0b9
+
0cb0b9
+	for (forbidden = atr_forbidden_sign; *forbidden; forbidden++) {
0cb0b9
+		if (sign == *forbidden)
0cb0b9
+			return 1;
0cb0b9
+	}
0cb0b9
+
0cb0b9
+	return 0;
0cb0b9
+}
0cb0b9
+
0cb0b9
+/* Updates table->checksum so the RS will be considered bootable (or not) */
0cb0b9
+static void
0cb0b9
+atr_table_set_boot (AtariRawTable* table, int boot)
0cb0b9
+{
0cb0b9
+	uint16_t boot_cksum, noboot_cksum;
0cb0b9
+	uint16_t sum;
0cb0b9
+
0cb0b9
+	table->checksum = 0;
0cb0b9
+	sum = atr_calc_rs_sum (table);
0cb0b9
+	boot_cksum = BOOTABLE_CKSUM - sum;
0cb0b9
+
0cb0b9
+	if (boot) {
0cb0b9
+		table->checksum = PED_CPU_TO_BE16 (boot_cksum);
0cb0b9
+		return;
0cb0b9
+	}
0cb0b9
+
0cb0b9
+	noboot_cksum = NONBOOT_CKSUM - sum;
0cb0b9
+
0cb0b9
+	while (atr_sign_is_forbidden (noboot_cksum)
0cb0b9
+	       || noboot_cksum == boot_cksum)
0cb0b9
+		noboot_cksum++;
0cb0b9
+
0cb0b9
+	table->checksum = PED_CPU_TO_BE16 (noboot_cksum);
0cb0b9
+}
0cb0b9
+
0cb0b9
+/* Fill an used partition entry */
0cb0b9
+static void
0cb0b9
+atr_fill_raw_entry (AtariRawPartition* rawpart, uint8_t flag, const char* id,
0cb0b9
+		    uint32_t start, uint32_t size )
0cb0b9
+{
0cb0b9
+	rawpart->flag = PART_FLAG_USED | flag;
0cb0b9
+	atr_pid_assign (rawpart->id, id);
0cb0b9
+	rawpart->start = PED_CPU_TO_BE32 (start);
0cb0b9
+	rawpart->size = PED_CPU_TO_BE32 (size);
0cb0b9
+}
0cb0b9
+
0cb0b9
+static int
0cb0b9
+atr_write_logicals (const PedDisk* disk)
0cb0b9
+{
0cb0b9
+	AtariRawTable	table;
0cb0b9
+	PedPartition*	log_curr;
0cb0b9
+	PedPartition*	log_next;
0cb0b9
+	PedPartition*	ext;
0cb0b9
+	PedPartition*	part;
0cb0b9
+	PedSector	exts;
0cb0b9
+	PedSector	parts;
0cb0b9
+	AtariPart*	atr_part;
0cb0b9
+	int		first_log, pnum, i;
0cb0b9
+
0cb0b9
+	PED_ASSERT (disk != NULL);
0cb0b9
+
0cb0b9
+	ext = ped_disk_extended_partition (disk);
0cb0b9
+	exts = parts = ext->geom.start;
0cb0b9
+
0cb0b9
+	pnum = first_log = atr_find_first_log (disk);
0cb0b9
+
0cb0b9
+	while (1) {
0cb0b9
+		if (pnum != -1) {
0cb0b9
+			log_curr = ped_disk_get_partition (disk, pnum);
0cb0b9
+			log_next = ped_disk_get_partition (disk, pnum + 1);
0cb0b9
+		} else {
0cb0b9
+			log_curr = log_next = NULL;
0cb0b9
+		}
0cb0b9
+
0cb0b9
+		if (log_curr && !(log_curr->type & PED_PARTITION_LOGICAL))
0cb0b9
+			log_curr = NULL;
0cb0b9
+		if (log_next && !(log_next->type & PED_PARTITION_LOGICAL))
0cb0b9
+			log_next = NULL;
0cb0b9
+
0cb0b9
+		PED_ASSERT (pnum == first_log || log_curr);
0cb0b9
+
0cb0b9
+		part = ped_disk_get_partition_by_sector (disk, parts);
0cb0b9
+		if (part && ped_partition_is_active (part)) {
0cb0b9
+			if (log_curr)
0cb0b9
+				ped_exception_throw (
0cb0b9
+					PED_EXCEPTION_ERROR,
0cb0b9
+					PED_EXCEPTION_CANCEL,
0cb0b9
+					_("No room at sector %lli to store ARS "
0cb0b9
+					  "of logical partition %d."),
0cb0b9
+					parts, pnum );
0cb0b9
+			else
0cb0b9
+				ped_exception_throw (
0cb0b9
+					PED_EXCEPTION_ERROR,
0cb0b9
+					PED_EXCEPTION_CANCEL,
0cb0b9
+				      _("No room at sector %lli to store ARS."),
0cb0b9
+					parts );
0cb0b9
+			return 0;
0cb0b9
+		}
0cb0b9
+
0cb0b9
+		if (!ped_device_read (disk->dev, &table, parts, 1))
0cb0b9
+			return 0;
0cb0b9
+
0cb0b9
+		if (!log_curr) {
0cb0b9
+			PED_ASSERT (!log_next);
0cb0b9
+
0cb0b9
+			for (i = 0; i < N_AHDI; i++)
0cb0b9
+				table.part[i].flag &= ~PART_FLAG_USED;
0cb0b9
+		} else {
0cb0b9
+			atr_part = ATARI_PART (log_curr);
0cb0b9
+			atr_fill_raw_entry (&table.part[0], atr_part->flag,
0cb0b9
+					    atr_part->part_id,
0cb0b9
+					    log_curr->geom.start - parts,
0cb0b9
+					    log_curr->geom.length );
0cb0b9
+
0cb0b9
+			for (i = 1; i < N_AHDI; i++)
0cb0b9
+				table.part[i].flag &= ~PART_FLAG_USED;
0cb0b9
+
0cb0b9
+			if (log_next) {
0cb0b9
+				atr_fill_raw_entry (&table.part[1], 0, "XGM",
0cb0b9
+					log_next->geom.start - 1 - exts,
0cb0b9
+					log_next->geom.length + 1 );
0cb0b9
+			}
0cb0b9
+		}
0cb0b9
+
0cb0b9
+		/* TODO: check if we can set that bootable, and when */
0cb0b9
+		atr_table_set_boot (&table, 0);
0cb0b9
+
0cb0b9
+		if (!ped_device_write (disk->dev, &table, parts, 1))
0cb0b9
+			return 0;
0cb0b9
+
0cb0b9
+		if (!log_next)
0cb0b9
+			break;
0cb0b9
+
0cb0b9
+		parts = log_next->geom.start - 1;
0cb0b9
+		pnum++;
0cb0b9
+	}
0cb0b9
+
0cb0b9
+	return 1;
0cb0b9
+}
0cb0b9
+
0cb0b9
+static int _GL_ATTRIBUTE_PURE
0cb0b9
+_disk_logical_partition_count (const PedDisk* disk)
0cb0b9
+{
0cb0b9
+	PedPartition*	walk;
0cb0b9
+
0cb0b9
+	int		count = 0;
0cb0b9
+
0cb0b9
+	PED_ASSERT (disk != NULL);
0cb0b9
+	for (walk = disk->part_list; walk;
0cb0b9
+	     walk = ped_disk_next_partition (disk, walk)) {
0cb0b9
+		if (ped_partition_is_active (walk)
0cb0b9
+		    && (walk->type & PED_PARTITION_LOGICAL))
0cb0b9
+			count++;
0cb0b9
+	}
0cb0b9
+
0cb0b9
+	return count;
0cb0b9
+}
0cb0b9
+
0cb0b9
+/* Load the HD size from the table and ask to fix it if != device size. */
0cb0b9
+static int
0cb0b9
+atr_load_fix_hdsize (const PedDisk* disk, uint32_t* rs_hd_size, AtariRawTable* table)
0cb0b9
+{
0cb0b9
+	AtariDisk*	atr_disk = ATARI_DISK (disk);
0cb0b9
+	int		result = PED_EXCEPTION_UNHANDLED;
0cb0b9
+
0cb0b9
+	*rs_hd_size = PED_BE32_TO_CPU (table->hd_size);
0cb0b9
+	if (*rs_hd_size != disk->dev->length) {
0cb0b9
+		if (atr_disk->has_been_read) {
0cb0b9
+			result = ped_exception_throw (
0cb0b9
+				PED_EXCEPTION_WARNING,
0cb0b9
+				PED_EXCEPTION_FIX | PED_EXCEPTION_IGNORE_CANCEL,
0cb0b9
+				_("The sector count that is stored in the "
0cb0b9
+				  "partition table does not correspond "
0cb0b9
+				  "to the size of your device.  Do you "
0cb0b9
+				  "want to fix the partition table?") );
0cb0b9
+			if (result == PED_EXCEPTION_CANCEL)
0cb0b9
+				return 0;
0cb0b9
+		}
0cb0b9
+
0cb0b9
+		if (result == PED_EXCEPTION_UNHANDLED)
0cb0b9
+			result = PED_EXCEPTION_FIX;
0cb0b9
+
0cb0b9
+		if (result == PED_EXCEPTION_FIX) {
0cb0b9
+			*rs_hd_size = disk->dev->length;
0cb0b9
+			table->hd_size = PED_CPU_TO_BE32(*rs_hd_size);
0cb0b9
+		}
0cb0b9
+	}
0cb0b9
+	return 1;
0cb0b9
+}
0cb0b9
+
0cb0b9
+/* Try to init the HDX compatibility Bad Sectors List. */
0cb0b9
+static int
0cb0b9
+atr_empty_init_bsl (const PedDisk* disk)
0cb0b9
+{
0cb0b9
+	uint8_t		zeros[PED_SECTOR_SIZE_DEFAULT];
0cb0b9
+	PedSector	sec;
0cb0b9
+	PedPartition*	part;
0cb0b9
+	AtariDisk*	atr_disk = ATARI_DISK (disk);
0cb0b9
+
0cb0b9
+	memset (zeros, 0, PED_SECTOR_SIZE_DEFAULT);
0cb0b9
+	for (sec = atr_disk->bsl_start;
0cb0b9
+	     sec < atr_disk->bsl_start + atr_disk->bsl_count;
0cb0b9
+	     sec++ ) {
0cb0b9
+		if (sec == atr_disk->bsl_start)
0cb0b9
+			zeros[3] = 0xA5;
0cb0b9
+		else
0cb0b9
+			zeros[3] = 0;
0cb0b9
+		part = ped_disk_get_partition_by_sector (disk, sec);
0cb0b9
+		if (part && ped_partition_is_active (part)) {
0cb0b9
+			ped_exception_throw (
0cb0b9
+				PED_EXCEPTION_ERROR,
0cb0b9
+				PED_EXCEPTION_CANCEL,
0cb0b9
+				_("No room at sector %lli to store BSL."),
0cb0b9
+				sec );
0cb0b9
+			return 0;
0cb0b9
+		}
0cb0b9
+		ped_device_write (disk->dev, zeros, sec, 1);
0cb0b9
+	}
0cb0b9
+	atr_disk->HDX_comp = 0;
0cb0b9
+	return 1;
0cb0b9
+}
0cb0b9
+
0cb0b9
+static int
0cb0b9
+atari_write (const PedDisk* disk)
0cb0b9
+{
0cb0b9
+	AtariRawTable	table;
0cb0b9
+	AtariDisk*	atr_disk;
0cb0b9
+	AtariPart*	atr_part;
0cb0b9
+	PedPartition*	log;
0cb0b9
+	PedPartition*	ext_part;
0cb0b9
+	PedPartition*	part = NULL;
0cb0b9
+	uint32_t	rs_hd_size;
0cb0b9
+	int		i, xgm_begin, pnum, append_ext;
0cb0b9
+	int		put_sign, boot, prim_count, last_num;
0cb0b9
+	PED_ASSERT (disk != NULL);
0cb0b9
+	PED_ASSERT (disk->dev != NULL);
0cb0b9
+	atr_disk = ATARI_DISK (disk);
0cb0b9
+	PED_ASSERT (atr_disk != NULL);
0cb0b9
+
0cb0b9
+	prim_count = ped_disk_get_primary_partition_count (disk);
0cb0b9
+	last_num = ped_disk_get_last_partition_num (disk);
0cb0b9
+	ext_part = ped_disk_extended_partition (disk);
0cb0b9
+
0cb0b9
+	/* WARNING: similar/related code in atari_enumerate */
0cb0b9
+	xgm_begin = ((log = ped_disk_get_partition (disk, 1))
0cb0b9
+		      && (log->type & PED_PARTITION_LOGICAL));
0cb0b9
+	PED_ASSERT (atr_disk->format != FMT_ICD || ext_part == NULL);
0cb0b9
+	PED_ASSERT (atr_disk->format != FMT_XGM || prim_count + xgm_begin <= N_AHDI);
0cb0b9
+	PED_ASSERT (atr_disk->format != FMT_AHDI || (ext_part == NULL && prim_count + xgm_begin <= N_AHDI));
0cb0b9
+
0cb0b9
+	/* Device Spec ok for Atari label? */
0cb0b9
+	if (!atr_can_use_dev (disk->dev))
0cb0b9
+		goto error;
0cb0b9
+
0cb0b9
+	if (!ped_device_read (disk->dev, (void*) &table, 0, 1))
0cb0b9
+		goto error;
0cb0b9
+
0cb0b9
+	boot = atr_is_boot_table (&table);
0cb0b9
+
0cb0b9
+	table.bsl_start = PED_CPU_TO_BE32 (atr_disk->bsl_start);
0cb0b9
+	table.bsl_count = PED_CPU_TO_BE32 (atr_disk->bsl_count);
0cb0b9
+
0cb0b9
+	/* Before anything else check the sector count and */
0cb0b9
+	/* fix it if necessary */
0cb0b9
+	if (!atr_load_fix_hdsize (disk, &rs_hd_size, &table))
0cb0b9
+		goto error;
0cb0b9
+
0cb0b9
+	append_ext =    (ext_part != NULL)
0cb0b9
+		     && (_disk_logical_partition_count (disk) == 0);
0cb0b9
+
0cb0b9
+	/* Fill the AHDI table */
0cb0b9
+	put_sign = (prim_count == 0);
0cb0b9
+	pnum = 1;
0cb0b9
+	for (i = 0; i < N_AHDI; i++) {
0cb0b9
+		if (pnum > last_num)
0cb0b9
+			part = NULL;
0cb0b9
+		else while (pnum <= last_num
0cb0b9
+			    && !(part = ped_disk_get_partition (disk, pnum)))
0cb0b9
+			pnum++;
0cb0b9
+
0cb0b9
+		if (put_sign) {
0cb0b9
+			atr_put_signature_entry (&table.part[i]);
0cb0b9
+			continue;
0cb0b9
+		}
0cb0b9
+
0cb0b9
+		if (!part && i != 0 && append_ext) {
0cb0b9
+			part = ext_part;
0cb0b9
+			append_ext = 0;
0cb0b9
+		}
0cb0b9
+
0cb0b9
+		if (!part || (i == 0 && xgm_begin)) {
0cb0b9
+			table.part[i].flag &= ~PART_FLAG_USED;
0cb0b9
+			continue;
0cb0b9
+		}
0cb0b9
+
0cb0b9
+		if (part->type & PED_PARTITION_LOGICAL)
0cb0b9
+			part = ext_part;
0cb0b9
+
0cb0b9
+		PED_ASSERT (part != NULL);
0cb0b9
+
0cb0b9
+		atr_part = ATARI_PART (part);
0cb0b9
+		atr_fill_raw_entry (&table.part[i], atr_part->flag,
0cb0b9
+				    atr_part->part_id, part->geom.start,
0cb0b9
+				    part->geom.length );
0cb0b9
+
0cb0b9
+		if (part->type & PED_PARTITION_EXTENDED) {
0cb0b9
+			while (pnum <= last_num) {
0cb0b9
+				part = ped_disk_get_partition (disk, pnum);
0cb0b9
+				if (part &&
0cb0b9
+				    !(part->type & PED_PARTITION_LOGICAL))
0cb0b9
+					break;
0cb0b9
+				pnum++;
0cb0b9
+			}
0cb0b9
+		} else
0cb0b9
+			pnum++;
0cb0b9
+	}
0cb0b9
+
0cb0b9
+	if ((ext_part != NULL || atr_disk->format == FMT_AHDI)
0cb0b9
+	    && pnum <= last_num) {
0cb0b9
+		ped_exception_throw (PED_EXCEPTION_BUG, PED_EXCEPTION_CANCEL,
0cb0b9
+			_("There were remaining partitions after filling "
0cb0b9
+			  "the main AHDI table.") );
0cb0b9
+		goto error;
0cb0b9
+	}
0cb0b9
+
0cb0b9
+	/* Leave XGM or ICD mode if uneeded */
0cb0b9
+	if (pnum > last_num
0cb0b9
+	    && (atr_disk->format == FMT_ICD || ext_part == NULL))
0cb0b9
+		atr_disk->format = FMT_AHDI;
0cb0b9
+
0cb0b9
+	/* If AHDI mode, check that no ICD will be detected */
0cb0b9
+	/* and propose to fix */
0cb0b9
+	if (atr_disk->format == FMT_AHDI
0cb0b9
+	    && atr_part_valid (&table.icd_part[0], rs_hd_size)
0cb0b9
+	    && atr_part_known (&table.icd_part[0], atr_known_icd_pid)) {
0cb0b9
+		int result = PED_EXCEPTION_UNHANDLED;
0cb0b9
+		result = ped_exception_throw (
0cb0b9
+			PED_EXCEPTION_WARNING,
0cb0b9
+			PED_EXCEPTION_YES_NO_CANCEL,
0cb0b9
+			_("The main AHDI table has been filled with all "
0cb0b9
+			  "partitions but the ICD table is not empty "
0cb0b9
+			  "so more partitions of unknown size and position "
0cb0b9
+			  "will be detected by ICD compatible software.  Do "
0cb0b9
+			  "you want to invalidate the ICD table?") );
0cb0b9
+		if (result == PED_EXCEPTION_YES
0cb0b9
+		    || result == PED_EXCEPTION_UNHANDLED)
0cb0b9
+			table.icd_part[0].flag &= ~PART_FLAG_USED;
0cb0b9
+		else if (result == PED_EXCEPTION_CANCEL)
0cb0b9
+			goto error;
0cb0b9
+	}
0cb0b9
+
0cb0b9
+	if (put_sign)
0cb0b9
+		goto write_to_dev;
0cb0b9
+
0cb0b9
+	/* Fill the ICD table */
0cb0b9
+	if (atr_disk->format == FMT_ICD)
0cb0b9
+	for (i = 0; i < N_ICD; i++) {
0cb0b9
+		if (pnum > last_num)
0cb0b9
+			part = NULL;
0cb0b9
+		else while (pnum <= last_num
0cb0b9
+			    && !(part = ped_disk_get_partition (disk, pnum)))
0cb0b9
+			pnum++;
0cb0b9
+
0cb0b9
+		if (!part) {
0cb0b9
+			table.icd_part[i].flag &= ~PART_FLAG_USED;
0cb0b9
+			continue;
0cb0b9
+		}
0cb0b9
+
0cb0b9
+		if (part->type & PED_PARTITION_EXTENDED
0cb0b9
+		    || part->type & PED_PARTITION_LOGICAL) {
0cb0b9
+			ped_exception_throw (
0cb0b9
+				PED_EXCEPTION_BUG,
0cb0b9
+				PED_EXCEPTION_CANCEL,
0cb0b9
+				_("ICD entries can't contain extended or "
0cb0b9
+				  "logical partitions.") );
0cb0b9
+			goto error;
0cb0b9
+		}
0cb0b9
+
0cb0b9
+		atr_part = ATARI_PART (part);
0cb0b9
+		atr_fill_raw_entry (&table.icd_part[i], atr_part->flag,
0cb0b9
+				    atr_part->icd_id, part->geom.start,
0cb0b9
+				    part->geom.length );
0cb0b9
+
0cb0b9
+		pnum++;
0cb0b9
+	}
0cb0b9
+
0cb0b9
+	/* Write the chained list of logical partitions */
0cb0b9
+	if (atr_disk->format == FMT_XGM) {
0cb0b9
+		if (!atr_write_logicals (disk))
0cb0b9
+			goto error;
0cb0b9
+	}
0cb0b9
+
0cb0b9
+write_to_dev:
0cb0b9
+	if (pnum <= last_num) {
0cb0b9
+		ped_exception_throw (PED_EXCEPTION_BUG, PED_EXCEPTION_CANCEL,
0cb0b9
+			_("There were remaining partitions after filling "
0cb0b9
+			  "the tables.") );
0cb0b9
+		goto error;
0cb0b9
+	}
0cb0b9
+
0cb0b9
+	/* Do we need to do that in case of failure too??? */
0cb0b9
+	atr_table_set_boot (&table, boot);
0cb0b9
+
0cb0b9
+	/* Commit the root sector... */
0cb0b9
+	if (!ped_device_write (disk->dev, (void*) &table, 0, 1)
0cb0b9
+	    || !ped_device_sync (disk->dev))
0cb0b9
+		goto error;
0cb0b9
+
0cb0b9
+	/* Try to init the HDX compatibility Bad Sectors List if needed. */
0cb0b9
+	if (atr_disk->HDX_comp && !atr_empty_init_bsl (disk))
0cb0b9
+		goto error;
0cb0b9
+
0cb0b9
+	atr_disk->has_been_read = 1;
0cb0b9
+	return ped_device_sync (disk->dev);
0cb0b9
+
0cb0b9
+error:
0cb0b9
+	atr_disk->has_been_read = 0;
0cb0b9
+	return 0;
0cb0b9
+}
0cb0b9
+#endif
0cb0b9
+
0cb0b9
+/* If extended partition in ICD mode, generate an error and returns 1 */
0cb0b9
+/* else returns 0 */
0cb0b9
+static int
0cb0b9
+atr_xgm_in_icd (const PedDisk* disk, PedPartitionType part_type)
0cb0b9
+{
0cb0b9
+	AtariDisk* atrdisk;
0cb0b9
+
0cb0b9
+	PED_ASSERT (disk != NULL);
0cb0b9
+
0cb0b9
+	if (part_type & PED_PARTITION_EXTENDED) {
0cb0b9
+		atrdisk = ATARI_DISK (disk);
0cb0b9
+		if (atrdisk->format == FMT_ICD) {
0cb0b9
+			ped_exception_throw (
0cb0b9
+			      PED_EXCEPTION_ERROR, PED_EXCEPTION_CANCEL,
0cb0b9
+			      _("You can't use an extended XGM partition in "
0cb0b9
+				"ICD mode (more than %d primary partitions, if "
0cb0b9
+				"XGM is the first one it counts for two)."),
0cb0b9
+				N_AHDI );
0cb0b9
+			return 1;
0cb0b9
+		}
0cb0b9
+	}
0cb0b9
+
0cb0b9
+	return 0;
0cb0b9
+}
0cb0b9
+
0cb0b9
+static PedPartition*
0cb0b9
+atari_partition_new (const PedDisk* disk, PedPartitionType part_type,
0cb0b9
+		     const PedFileSystemType* fs_type,
0cb0b9
+		     PedSector start, PedSector end)
0cb0b9
+{
0cb0b9
+	PedPartition*	part;
0cb0b9
+	AtariPart*	atrpart;
0cb0b9
+
0cb0b9
+	if (atr_xgm_in_icd(disk, part_type))
0cb0b9
+		return 0;
0cb0b9
+
0cb0b9
+	part = _ped_partition_alloc (disk, part_type, fs_type, start, end);
0cb0b9
+	if (!part)
0cb0b9
+		goto error;
0cb0b9
+	if (ped_partition_is_active (part)) {
0cb0b9
+		part->disk_specific = atrpart = ped_malloc (sizeof (AtariPart));
0cb0b9
+		if (!atrpart)
0cb0b9
+			goto error_free_part;
0cb0b9
+		memset (atrpart, 0, sizeof (AtariPart));
0cb0b9
+	} else {
0cb0b9
+		part->disk_specific = NULL;
0cb0b9
+	}
0cb0b9
+	return part;
0cb0b9
+
0cb0b9
+error_free_part:
0cb0b9
+	_ped_partition_free (part);
0cb0b9
+error:
0cb0b9
+	return NULL;
0cb0b9
+}
0cb0b9
+
0cb0b9
+static PedPartition*
0cb0b9
+atari_partition_duplicate (const PedPartition* part)
0cb0b9
+{
0cb0b9
+	PedPartition*	new_part;
0cb0b9
+
0cb0b9
+	new_part = ped_partition_new (part->disk, part->type,
0cb0b9
+				      part->fs_type, part->geom.start,
0cb0b9
+				      part->geom.end);
0cb0b9
+	if (!new_part)
0cb0b9
+		return NULL;
0cb0b9
+	new_part->num = part->num;
0cb0b9
+	if (ped_partition_is_active (part))
0cb0b9
+		memcpy (new_part->disk_specific, part->disk_specific,
0cb0b9
+			sizeof (AtariPart));
0cb0b9
+
0cb0b9
+	return new_part;
0cb0b9
+}
0cb0b9
+
0cb0b9
+static void
0cb0b9
+atari_partition_destroy (PedPartition* part)
0cb0b9
+{
0cb0b9
+	PED_ASSERT (part != NULL);
0cb0b9
+
0cb0b9
+	if (ped_partition_is_active (part)) {
0cb0b9
+		PED_ASSERT (part->disk_specific != NULL);
0cb0b9
+		free (part->disk_specific);
0cb0b9
+	}
0cb0b9
+	_ped_partition_free (part);
0cb0b9
+}
0cb0b9
+
0cb0b9
+/* Note: fs_type is NULL for extended partitions */
0cb0b9
+static int
0cb0b9
+atari_partition_set_system (PedPartition* part,
0cb0b9
+			    const PedFileSystemType* fs_type)
0cb0b9
+{
0cb0b9
+	AtariPart*	atrpart;
0cb0b9
+	AtariFS2PartId* fs2id;
0cb0b9
+	PED_ASSERT (part != NULL);
0cb0b9
+	atrpart = ATARI_PART (part);
0cb0b9
+	PED_ASSERT (atrpart != NULL);
0cb0b9
+
0cb0b9
+	part->fs_type = fs_type;
0cb0b9
+
0cb0b9
+	if (atr_xgm_in_icd(part->disk, part->type))
0cb0b9
+		return 0;
0cb0b9
+
0cb0b9
+	if (part->type & PED_PARTITION_EXTENDED) {
0cb0b9
+		strcpy (atrpart->part_id, "XGM");
0cb0b9
+		strcpy (atrpart->icd_id,  "XGM");
0cb0b9
+		return 1;
0cb0b9
+	}
0cb0b9
+
0cb0b9
+	if (!fs_type) {
0cb0b9
+		strcpy (atrpart->part_id, "RAW");
0cb0b9
+		strcpy (atrpart->icd_id,  "RAW");
0cb0b9
+		return 1;
0cb0b9
+	}
0cb0b9
+
0cb0b9
+	for (fs2id = atr_fs2pid; fs2id->fs; fs2id++) {
0cb0b9
+		if (!*fs2id->fs    /* default entry */
0cb0b9
+		    || ((!strcmp (fs_type->name, fs2id->fs)
0cb0b9
+		        && part->geom.length < fs2id->max_sectors))) {
0cb0b9
+
0cb0b9
+			strcpy (atrpart->part_id, fs2id->pid);
0cb0b9
+			if (atr_pid_known (fs2id->pid, atr_known_icd_pid))
0cb0b9
+				strcpy (atrpart->icd_id, fs2id->pid);
0cb0b9
+			else
0cb0b9
+				strcpy (atrpart->icd_id, "RAW");
0cb0b9
+
0cb0b9
+			break;
0cb0b9
+		}
0cb0b9
+	}
0cb0b9
+	PED_ASSERT (fs2id->fs != NULL);
0cb0b9
+
0cb0b9
+	return 1;
0cb0b9
+}
0cb0b9
+
0cb0b9
+static int
0cb0b9
+atari_partition_set_flag (PedPartition* part, PedPartitionFlag flag, int state)
0cb0b9
+{
0cb0b9
+	AtariPart* atr_part;
0cb0b9
+	AtariPartID2BootFlag* bf;
0cb0b9
+
0cb0b9
+	PED_ASSERT (part != NULL);
0cb0b9
+	atr_part = ATARI_PART (part);
0cb0b9
+	PED_ASSERT (atr_part != NULL);
0cb0b9
+
0cb0b9
+	if (flag != PED_PARTITION_BOOT)
0cb0b9
+		return 0;
0cb0b9
+
0cb0b9
+	if (state == 0) {
0cb0b9
+		atr_part->flag = 0;
0cb0b9
+	} else {
0cb0b9
+		for (bf = atr_pid2bf; *bf->pid; bf++) {
0cb0b9
+			if (atr_pid_eq (bf->pid, atr_part->part_id))
0cb0b9
+				break;
0cb0b9
+		}
0cb0b9
+		atr_part->flag = bf->flag;
0cb0b9
+	}
0cb0b9
+
0cb0b9
+	return 1;
0cb0b9
+}
0cb0b9
+
0cb0b9
+static int _GL_ATTRIBUTE_PURE
0cb0b9
+atari_partition_get_flag (const PedPartition* part, PedPartitionFlag flag)
0cb0b9
+{
0cb0b9
+	AtariPart* atr_part;
0cb0b9
+
0cb0b9
+	PED_ASSERT (part != NULL);
0cb0b9
+	atr_part = ATARI_PART (part);
0cb0b9
+	PED_ASSERT (atr_part != NULL);
0cb0b9
+
0cb0b9
+	if (flag != PED_PARTITION_BOOT)
0cb0b9
+		return 0;
0cb0b9
+
0cb0b9
+	return (atr_part->flag != 0);
0cb0b9
+}
0cb0b9
+
0cb0b9
+static int
0cb0b9
+atari_partition_is_flag_available (const PedPartition* part,
0cb0b9
+				   PedPartitionFlag flag)
0cb0b9
+{
0cb0b9
+	if (flag == PED_PARTITION_BOOT)
0cb0b9
+		return 1;
0cb0b9
+
0cb0b9
+	return 0;
0cb0b9
+}
0cb0b9
+
0cb0b9
+/* Adapted from disk_dos */
0cb0b9
+static PedConstraint*
0cb0b9
+atr_log_constraint (const PedPartition* part)
0cb0b9
+{
0cb0b9
+	const PedGeometry*	geom = &part->geom;
0cb0b9
+	PedGeometry	safe_space;
0cb0b9
+	PedSector	min_start;
0cb0b9
+	PedSector	max_end;
0cb0b9
+	PedDisk*	disk;
0cb0b9
+	PedDevice*	dev;
0cb0b9
+	PedPartition*	ext_part;
0cb0b9
+	PedPartition*	walk;
0cb0b9
+	int		first_log, not_first;
0cb0b9
+
0cb0b9
+	PED_ASSERT (part->disk != NULL);
0cb0b9
+	PED_ASSERT (part->disk->dev != NULL);
0cb0b9
+	ext_part = ped_disk_extended_partition (part->disk);
0cb0b9
+	PED_ASSERT (ext_part != NULL);
0cb0b9
+
0cb0b9
+	dev = (disk = part->disk) -> dev;
0cb0b9
+
0cb0b9
+	first_log = atr_find_first_log (disk);
0cb0b9
+	if (first_log == -1)
0cb0b9
+		first_log = part->num;
0cb0b9
+
0cb0b9
+	not_first = (part->num != first_log);
0cb0b9
+
0cb0b9
+	walk = ext_part->part_list;
0cb0b9
+
0cb0b9
+	min_start = ext_part->geom.start + 1 + not_first;
0cb0b9
+	max_end = ext_part->geom.end;
0cb0b9
+
0cb0b9
+	while (walk != NULL
0cb0b9
+		&& (   walk->geom.start - (walk->num != first_log)
0cb0b9
+						< geom->start - not_first
0cb0b9
+		    || walk->geom.start - (walk->num != first_log)
0cb0b9
+						< min_start ) ) {
0cb0b9
+		if (walk != part && ped_partition_is_active (walk))
0cb0b9
+			min_start = walk->geom.end + 1 + not_first;
0cb0b9
+		walk = walk->next;
0cb0b9
+	}
0cb0b9
+
0cb0b9
+	while (walk && (walk == part || !ped_partition_is_active (walk)))
0cb0b9
+		walk = walk->next;
0cb0b9
+
0cb0b9
+	if (walk)
0cb0b9
+		max_end = walk->geom.start - 1 - (walk->num != first_log);
0cb0b9
+
0cb0b9
+	if (min_start >= max_end)
0cb0b9
+		return NULL;
0cb0b9
+
0cb0b9
+	ped_geometry_init (&safe_space, dev, min_start,
0cb0b9
+			   max_end - min_start + 1);
0cb0b9
+	return ped_constraint_new_from_max (&safe_space);
0cb0b9
+}
0cb0b9
+
0cb0b9
+/* Adapted from disk_dos */
0cb0b9
+static PedGeometry*
0cb0b9
+art_min_extended_geom (const PedPartition* ext_part)
0cb0b9
+{
0cb0b9
+	PedDisk*	disk = ext_part->disk;
0cb0b9
+	PedPartition*	walk;
0cb0b9
+	PedGeometry*	min_geom;
0cb0b9
+	int		first_log;
0cb0b9
+
0cb0b9
+	first_log = atr_find_first_log (disk);
0cb0b9
+	if (first_log == -1)
0cb0b9
+		return NULL;
0cb0b9
+
0cb0b9
+	walk = ped_disk_get_partition (disk, first_log);
0cb0b9
+	PED_ASSERT (walk->type & PED_PARTITION_LOGICAL);
0cb0b9
+	min_geom = ped_geometry_duplicate (&walk->geom);
0cb0b9
+	if (!min_geom)
0cb0b9
+		return NULL;
0cb0b9
+	ped_geometry_set_start (min_geom, walk->geom.start - 1);
0cb0b9
+
0cb0b9
+	for (walk = ext_part->part_list; walk; walk = walk->next) {
0cb0b9
+		if (!ped_partition_is_active (walk) || walk->num == first_log)
0cb0b9
+			continue;
0cb0b9
+		if (walk->geom.start < min_geom->start)
0cb0b9
+			ped_geometry_set_start (min_geom, walk->geom.start - 2);
0cb0b9
+		if (walk->geom.end > min_geom->end)
0cb0b9
+			ped_geometry_set_end (min_geom, walk->geom.end);
0cb0b9
+	}
0cb0b9
+
0cb0b9
+	return min_geom;
0cb0b9
+}
0cb0b9
+
0cb0b9
+/* Adapted from disk_dos */
0cb0b9
+static PedConstraint*
0cb0b9
+atr_ext_constraint (const PedPartition* part)
0cb0b9
+{
0cb0b9
+	PedGeometry	start_range;
0cb0b9
+	PedGeometry	end_range;
0cb0b9
+	PedConstraint*	constraint;
0cb0b9
+	PedDevice*	dev;
0cb0b9
+	PedDisk*	disk;
0cb0b9
+	PedGeometry*	min;
0cb0b9
+
0cb0b9
+	PED_ASSERT (part->disk != NULL);
0cb0b9
+	PED_ASSERT (part->disk->dev != NULL);
0cb0b9
+
0cb0b9
+	dev = (disk = part->disk) -> dev;
0cb0b9
+	min = art_min_extended_geom (part);
0cb0b9
+
0cb0b9
+	if (min) {
0cb0b9
+		ped_geometry_init (&start_range, dev, 1, min->start);
0cb0b9
+		ped_geometry_init (&end_range, dev, min->end,
0cb0b9
+				   dev->length - min->end);
0cb0b9
+		ped_geometry_destroy (min);
0cb0b9
+	} else {
0cb0b9
+		ped_geometry_init (&start_range, dev, 1, dev->length - 1);
0cb0b9
+		ped_geometry_init (&end_range, dev, 1, dev->length - 1);
0cb0b9
+	}
0cb0b9
+
0cb0b9
+	constraint = ped_constraint_new (ped_alignment_any, ped_alignment_any,
0cb0b9
+				&start_range, &end_range, 1, dev->length);
0cb0b9
+	return constraint;
0cb0b9
+}
0cb0b9
+
0cb0b9
+static PedConstraint*
0cb0b9
+atr_prim_constraint (const PedPartition* part)
0cb0b9
+{
0cb0b9
+	PedDevice*	dev;
0cb0b9
+	PedGeometry	max;
0cb0b9
+
0cb0b9
+	PED_ASSERT (part->disk != NULL);
0cb0b9
+	PED_ASSERT (part->disk->dev != NULL);
0cb0b9
+
0cb0b9
+	dev = part->disk->dev;
0cb0b9
+
0cb0b9
+	ped_geometry_init (&max, dev, 1, dev->length - 1);
0cb0b9
+	return ped_constraint_new_from_max (&max;;
0cb0b9
+}
0cb0b9
+
0cb0b9
+/* inspiration from disk_dos */
0cb0b9
+static PedGeometry*
0cb0b9
+_best_solution (PedGeometry* a, PedGeometry* b)
0cb0b9
+{
0cb0b9
+	if (!a)
0cb0b9
+		return b;
0cb0b9
+	if (!b)
0cb0b9
+		return a;
0cb0b9
+
0cb0b9
+	if (a->length < b->length)
0cb0b9
+		goto choose_b;
0cb0b9
+
0cb0b9
+	ped_geometry_destroy (b);
0cb0b9
+	return a;
0cb0b9
+
0cb0b9
+choose_b:
0cb0b9
+	ped_geometry_destroy (a);
0cb0b9
+	return b;
0cb0b9
+}
0cb0b9
+
0cb0b9
+/* copied from disk_dos */
0cb0b9
+static PedGeometry*
0cb0b9
+_try_constraint (const PedPartition* part, const PedConstraint* external,
0cb0b9
+		 PedConstraint* internal)
0cb0b9
+{
0cb0b9
+	PedConstraint*		intersection;
0cb0b9
+	PedGeometry*		solution;
0cb0b9
+
0cb0b9
+	intersection = ped_constraint_intersect (external, internal);
0cb0b9
+	ped_constraint_destroy (internal);
0cb0b9
+	if (!intersection)
0cb0b9
+		return NULL;
0cb0b9
+
0cb0b9
+	solution = ped_constraint_solve_nearest (intersection, &part->geom);
0cb0b9
+	ped_constraint_destroy (intersection);
0cb0b9
+	return solution;
0cb0b9
+}
0cb0b9
+
0cb0b9
+/*
0cb0b9
+ * internal is either the primary or extented constraint.
0cb0b9
+ * If there's no BSL, the is the only internal constraint considered.
0cb0b9
+ * If there's a BSL, try to fit the partition before or after (and
0cb0b9
+ * choose the best fit, the one which results in the greatest size...)
0cb0b9
+ */
0cb0b9
+static int
0cb0b9
+atr_prim_align (PedPartition* part, const PedConstraint* constraint,
0cb0b9
+		PedConstraint* internal)
0cb0b9
+{
0cb0b9
+	PedDevice*	dev;
0cb0b9
+	AtariDisk*	atr_disk;
0cb0b9
+	PedConstraint*	cut;
0cb0b9
+	PedGeometry*	solution = NULL;
0cb0b9
+	PedGeometry	max;
0cb0b9
+	PedSector	bsl_end;
0cb0b9
+
0cb0b9
+	PED_ASSERT (part->disk != NULL);
0cb0b9
+	PED_ASSERT (part->disk->dev != NULL);
0cb0b9
+	dev = part->disk->dev;
0cb0b9
+	atr_disk = ATARI_DISK (part->disk);
0cb0b9
+	PED_ASSERT (atr_disk != NULL);
0cb0b9
+
0cb0b9
+	/* No BSL */
0cb0b9
+	if (!atr_disk->bsl_start && !atr_disk->bsl_count) {
0cb0b9
+		/* Note: _ped_partition_attempt_align will destroy internal */
0cb0b9
+		return _ped_partition_attempt_align(part, constraint, internal);
0cb0b9
+	}
0cb0b9
+
0cb0b9
+	/* BSL, try to fit before */
0cb0b9
+	if (atr_disk->bsl_start > 1) {
0cb0b9
+		ped_geometry_init (&max, dev, 1, atr_disk->bsl_start - 1);
0cb0b9
+		cut = ped_constraint_new_from_max (&max;;
0cb0b9
+		solution = _best_solution (solution,
0cb0b9
+				_try_constraint (part, constraint,
0cb0b9
+				     ped_constraint_intersect (internal, cut)));
0cb0b9
+		ped_constraint_destroy (cut);
0cb0b9
+	}
0cb0b9
+
0cb0b9
+	/* BSL, try to fit after, take the best solution */
0cb0b9
+	bsl_end = atr_disk->bsl_start + atr_disk->bsl_count;
0cb0b9
+	if (bsl_end < dev->length) {
0cb0b9
+		ped_geometry_init (&max, dev, bsl_end, dev->length - bsl_end);
0cb0b9
+		cut = ped_constraint_new_from_max (&max;;
0cb0b9
+		solution = _best_solution (solution,
0cb0b9
+				_try_constraint (part, constraint,
0cb0b9
+				     ped_constraint_intersect (internal, cut)));
0cb0b9
+		ped_constraint_destroy (cut);
0cb0b9
+	}
0cb0b9
+
0cb0b9
+	ped_constraint_destroy (internal);
0cb0b9
+
0cb0b9
+	if (solution) {
0cb0b9
+		ped_geometry_set (&part->geom, solution->start,
0cb0b9
+				  solution->length);
0cb0b9
+		ped_geometry_destroy (solution);
0cb0b9
+		return 1;
0cb0b9
+	}
0cb0b9
+
0cb0b9
+	return 0;
0cb0b9
+}
0cb0b9
+
0cb0b9
+static int
0cb0b9
+atari_partition_align (PedPartition* part, const PedConstraint* constraint)
0cb0b9
+{
0cb0b9
+	PED_ASSERT (part != NULL);
0cb0b9
+
0cb0b9
+	switch (part->type) {
0cb0b9
+	    case PED_PARTITION_LOGICAL:
0cb0b9
+		if (_ped_partition_attempt_align (part, constraint,
0cb0b9
+						  atr_log_constraint (part) ) )
0cb0b9
+			return 1;
0cb0b9
+		break;
0cb0b9
+	    case PED_PARTITION_EXTENDED:
0cb0b9
+		if (atr_prim_align (part, constraint,
0cb0b9
+				    atr_ext_constraint (part) ) )
0cb0b9
+			return 1;
0cb0b9
+		break;
0cb0b9
+	    default:
0cb0b9
+		if (atr_prim_align (part, constraint,
0cb0b9
+				    atr_prim_constraint (part) ) )
0cb0b9
+			return 1;
0cb0b9
+		break;
0cb0b9
+	}
0cb0b9
+
0cb0b9
+#ifndef DISCOVER_ONLY
0cb0b9
+	ped_exception_throw (
0cb0b9
+		PED_EXCEPTION_ERROR,
0cb0b9
+		PED_EXCEPTION_CANCEL,
0cb0b9
+		_("Unable to satisfy all constraints on the partition."));
0cb0b9
+#endif
0cb0b9
+	return 0;
0cb0b9
+}
0cb0b9
+
0cb0b9
+/* increment numbers of any non logical partition found after the last */
0cb0b9
+/* logical one, to make room for a new logical partition */
0cb0b9
+static int
0cb0b9
+art_room_for_logic (PedDisk* disk)
0cb0b9
+{
0cb0b9
+	PedPartition*	part;
0cb0b9
+	int		num, last_logic, last;
0cb0b9
+
0cb0b9
+	/* too many partitions ? */
0cb0b9
+	last = ped_disk_get_last_partition_num (disk);
0cb0b9
+	if (last >= MAXIMUM_PARTS)
0cb0b9
+		return 0;
0cb0b9
+
0cb0b9
+	/* find the last logical partition */
0cb0b9
+	last_logic = 0;
0cb0b9
+	for (num = 1; num <= last; num++) {
0cb0b9
+		part = ped_disk_get_partition (disk, num);
0cb0b9
+		if (part && ped_partition_is_active (part)
0cb0b9
+		         && (part->type & PED_PARTITION_LOGICAL))
0cb0b9
+			last_logic = num;
0cb0b9
+	}
0cb0b9
+
0cb0b9
+	if (!last_logic)
0cb0b9
+		return 1;
0cb0b9
+
0cb0b9
+	/* increment */
0cb0b9
+	for (num = last; num > last_logic; num--) {
0cb0b9
+		part = ped_disk_get_partition (disk, num);
0cb0b9
+		if (part && ped_partition_is_active (part)
0cb0b9
+		         && !(part->type & ( PED_PARTITION_LOGICAL
0cb0b9
+					   | PED_PARTITION_EXTENDED))
0cb0b9
+			 && part->num > 0 )
0cb0b9
+			part->num++;
0cb0b9
+	}
0cb0b9
+
0cb0b9
+	return 1;
0cb0b9
+}
0cb0b9
+
0cb0b9
+static int
0cb0b9
+atari_partition_enumerate (PedPartition* part)
0cb0b9
+{
0cb0b9
+	AtariDisk*	atrdisk;
0cb0b9
+	PedPartition*	ext_part;
0cb0b9
+	PedPartition*	log;
0cb0b9
+	int		i, want_icd, want_xgm, num_max, xgm_begin, prim_count;
0cb0b9
+
0cb0b9
+	PED_ASSERT (part != NULL);
0cb0b9
+	PED_ASSERT (part->disk != NULL);
0cb0b9
+	atrdisk = ATARI_DISK (part->disk);
0cb0b9
+	PED_ASSERT (atrdisk != NULL);
0cb0b9
+
0cb0b9
+	/* WARNING: some similar/related code in atari_write */
0cb0b9
+	/* This is quite a <hack> : this function is probably the only way   */
0cb0b9
+	/* to know something has been / is going to be modified in the table.*/
0cb0b9
+	/* So we detect the current operation mode (AHDI/XGM/ICD) and report */
0cb0b9
+	/* errors (in which case we refuse to operate...) */
0cb0b9
+
0cb0b9
+	prim_count = ped_disk_get_primary_partition_count (part->disk);
0cb0b9
+	ext_part = ped_disk_extended_partition (part->disk);
0cb0b9
+
0cb0b9
+	/* <hack in the hack> : we can't reorder (yet) , so if we begin with */
0cb0b9
+	/* XGM the first slot must be empty */
0cb0b9
+	xgm_begin = ((log = ped_disk_get_partition (part->disk, 1))
0cb0b9
+		      && (log->type & PED_PARTITION_LOGICAL))
0cb0b9
+		    || ((part->num == -1)
0cb0b9
+		        && (part->type & PED_PARTITION_LOGICAL)
0cb0b9
+			&& !ped_disk_get_partition (part->disk, 1));
0cb0b9
+	/* </hack in the hack> */
0cb0b9
+
0cb0b9
+	PED_ASSERT (atrdisk->format != FMT_ICD || ext_part == NULL);
0cb0b9
+	PED_ASSERT (atrdisk->format != FMT_XGM
0cb0b9
+		    || prim_count + xgm_begin <= N_AHDI);
0cb0b9
+	PED_ASSERT (atrdisk->format != FMT_AHDI
0cb0b9
+		    || (ext_part == NULL && prim_count + xgm_begin <= N_AHDI));
0cb0b9
+
0cb0b9
+	want_icd = ( ( prim_count
0cb0b9
+			+ xgm_begin
0cb0b9
+			+ ( (part->num == -1)
0cb0b9
+			    && !(part->type & PED_PARTITION_LOGICAL) ) )
0cb0b9
+		      > N_AHDI );
0cb0b9
+	want_xgm = ( (part->type & PED_PARTITION_EXTENDED)
0cb0b9
+	             || ext_part != NULL );
0cb0b9
+
0cb0b9
+	if (!want_xgm && !want_icd)
0cb0b9
+		atrdisk->format = FMT_AHDI;
0cb0b9
+	else if (want_xgm && !want_icd)
0cb0b9
+		atrdisk->format = FMT_XGM;
0cb0b9
+	else if (!want_xgm && want_icd)
0cb0b9
+		atrdisk->format = FMT_ICD;
0cb0b9
+	else {
0cb0b9
+		if (atr_xgm_in_icd (part->disk, PED_PARTITION_EXTENDED))
0cb0b9
+			return 0;
0cb0b9
+		else {
0cb0b9
+			ped_exception_throw (
0cb0b9
+				PED_EXCEPTION_ERROR, PED_EXCEPTION_CANCEL,
0cb0b9
+			      _("You can't use more than %d primary partitions "
0cb0b9
+			        "(ICD mode) if you use an extended XGM "
0cb0b9
+				"partition.  If XGM is the first partition "
0cb0b9
+				"it counts for two."),
0cb0b9
+				N_AHDI );
0cb0b9
+			return 0;
0cb0b9
+		}
0cb0b9
+	}
0cb0b9
+	/* End of </hack> */
0cb0b9
+
0cb0b9
+
0cb0b9
+	/* Ext will be numbered 0 and will stay 0... */
0cb0b9
+	if (part->num == 0)
0cb0b9
+		return 1;
0cb0b9
+
0cb0b9
+	if (part->num == -1) {
0cb0b9
+
0cb0b9
+		/* Linux don't show the ext part itself for Atari disk labels */
0cb0b9
+		/* so we use number 0 (could use a big number too, but that   */
0cb0b9
+		/* would be less cute ;) */
0cb0b9
+		if (part->type & PED_PARTITION_EXTENDED) {
0cb0b9
+			part->num = 0;
0cb0b9
+			return 1;
0cb0b9
+		}
0cb0b9
+
0cb0b9
+		switch (atrdisk->format) {
0cb0b9
+		    case FMT_AHDI:
0cb0b9
+		    case FMT_ICD:
0cb0b9
+			num_max = N_ICD + N_AHDI;
0cb0b9
+			break;
0cb0b9
+		    case FMT_XGM:
0cb0b9
+			num_max = MAXIMUM_PARTS;
0cb0b9
+			break;
0cb0b9
+		    default:
0cb0b9
+			num_max = 0;
0cb0b9
+			PED_ASSERT (0);
0cb0b9
+		}
0cb0b9
+
0cb0b9
+		/* make room for logical partitions */
0cb0b9
+		if (part->type & PED_PARTITION_LOGICAL) {
0cb0b9
+			if (!art_room_for_logic (part->disk))
0cb0b9
+				goto error_alloc_failed;
0cb0b9
+		}
0cb0b9
+
0cb0b9
+		/* find an unused number */
0cb0b9
+		for (i = 1; i <= num_max; i++) {
0cb0b9
+			if (!ped_disk_get_partition (part->disk, i)) {
0cb0b9
+				part->num = i;
0cb0b9
+				return 1;
0cb0b9
+			}
0cb0b9
+		}
0cb0b9
+
0cb0b9
+	} else {
0cb0b9
+		/* find an unused number before or don't re-number */
0cb0b9
+		for (i = 1; i < part->num; i++) {
0cb0b9
+			if (!ped_disk_get_partition (part->disk, i)) {
0cb0b9
+				part->num = i;
0cb0b9
+			}
0cb0b9
+		}
0cb0b9
+		return 1;
0cb0b9
+	}
0cb0b9
+
0cb0b9
+	/* failed to allocate a number */
0cb0b9
+error_alloc_failed:
0cb0b9
+#ifndef DISCOVER_ONLY
0cb0b9
+	ped_exception_throw (PED_EXCEPTION_ERROR, PED_EXCEPTION_CANCEL,
0cb0b9
+		_("Unable to allocate a partition number."));
0cb0b9
+#endif
0cb0b9
+	return 0;
0cb0b9
+}
0cb0b9
+
0cb0b9
+static int
0cb0b9
+atr_creat_add_metadata (PedDisk* disk, PedSector start, PedSector end,
0cb0b9
+			PedPartitionType type )
0cb0b9
+{
0cb0b9
+	PedPartition*	new_part;
0cb0b9
+	PedConstraint*	const_exact;
0cb0b9
+	int		added;
0cb0b9
+
0cb0b9
+	type |= PED_PARTITION_METADATA;
0cb0b9
+	new_part = ped_partition_new (disk, type, NULL, start, end);
0cb0b9
+	if (!new_part)
0cb0b9
+		goto error;
0cb0b9
+
0cb0b9
+	const_exact = ped_constraint_exact (&new_part->geom);
0cb0b9
+	added = ped_disk_add_partition (disk, new_part, const_exact);
0cb0b9
+	ped_constraint_destroy (const_exact);
0cb0b9
+	if (!added)
0cb0b9
+		goto error_destroy_part;
0cb0b9
+
0cb0b9
+	return 1;
0cb0b9
+
0cb0b9
+error_destroy_part:
0cb0b9
+	ped_partition_destroy (new_part);
0cb0b9
+error:
0cb0b9
+	return 0;
0cb0b9
+}
0cb0b9
+
0cb0b9
+static int
0cb0b9
+atari_alloc_metadata (PedDisk* disk)
0cb0b9
+{
0cb0b9
+	PedPartition*	ext;
0cb0b9
+	PedPartition*	log;
0cb0b9
+	AtariDisk*	atr_disk;
0cb0b9
+	int		i;
0cb0b9
+
0cb0b9
+	PED_ASSERT (disk != NULL);
0cb0b9
+	PED_ASSERT (disk->dev != NULL);
0cb0b9
+	atr_disk = ATARI_DISK (disk);
0cb0b9
+	PED_ASSERT (atr_disk != NULL);
0cb0b9
+
0cb0b9
+	/* allocate 1 sector for the disk label at the start */
0cb0b9
+	if (!atr_creat_add_metadata (disk, 0, 0, 0))
0cb0b9
+		return 0;
0cb0b9
+
0cb0b9
+	/* allocate the sectors containing the BSL */
0cb0b9
+	if (atr_disk->bsl_start || atr_disk->bsl_count) {
0cb0b9
+		if (!atr_creat_add_metadata (disk, atr_disk->bsl_start,
0cb0b9
+					     atr_disk->bsl_start
0cb0b9
+					      + atr_disk->bsl_count - 1, 0 ) )
0cb0b9
+			return 0;
0cb0b9
+	}
0cb0b9
+
0cb0b9
+	ext = ped_disk_extended_partition (disk);
0cb0b9
+	if (ext) {
0cb0b9
+		if (!atr_creat_add_metadata (disk, ext->geom.start,
0cb0b9
+					     ext->geom.start,
0cb0b9
+					     PED_PARTITION_LOGICAL ) )
0cb0b9
+			return 0;
0cb0b9
+
0cb0b9
+		/* Find the first logical part */
0cb0b9
+		for (i = 1; i <= ped_disk_get_last_partition_num (disk); i++)
0cb0b9
+			if ((log = ped_disk_get_partition (disk, i))
0cb0b9
+			    && (log->type & PED_PARTITION_LOGICAL))
0cb0b9
+				break;
0cb0b9
+
0cb0b9
+		for (log = ext->part_list; log; log = log->next) {
0cb0b9
+			if ((log->type & ( PED_PARTITION_METADATA
0cb0b9
+					 | PED_PARTITION_FREESPACE))
0cb0b9
+			    || log->num == i)
0cb0b9
+				continue;
0cb0b9
+
0cb0b9
+			if (!atr_creat_add_metadata (disk, log->geom.start-1,
0cb0b9
+						     log->geom.start-1,
0cb0b9
+						     PED_PARTITION_LOGICAL ) )
0cb0b9
+				return 0;
0cb0b9
+		}
0cb0b9
+	}
0cb0b9
+
0cb0b9
+	return 1;
0cb0b9
+}
0cb0b9
+
0cb0b9
+static int _GL_ATTRIBUTE_PURE
0cb0b9
+atari_get_max_primary_partition_count (const PedDisk* disk)
0cb0b9
+{
0cb0b9
+	AtariDisk*	atr_disk;
0cb0b9
+
0cb0b9
+	PED_ASSERT (disk != NULL);
0cb0b9
+	atr_disk = ATARI_DISK (disk);
0cb0b9
+	PED_ASSERT (atr_disk != NULL);
0cb0b9
+
0cb0b9
+	return atr_disk->format == FMT_XGM ? N_AHDI : N_AHDI + N_ICD;
0cb0b9
+}
0cb0b9
+
0cb0b9
+static bool
0cb0b9
+atari_get_max_supported_partition_count (const PedDisk* disk, int *max_n)
0cb0b9
+{
0cb0b9
+	AtariDisk*	atr_disk;
0cb0b9
+
0cb0b9
+	PED_ASSERT (disk != NULL);
0cb0b9
+	atr_disk = ATARI_DISK (disk);
0cb0b9
+	PED_ASSERT (atr_disk != NULL);
0cb0b9
+
0cb0b9
+	*max_n = atr_disk->format == FMT_XGM ? N_AHDI : N_AHDI + N_ICD;
0cb0b9
+        return true;
0cb0b9
+}
0cb0b9
+
0cb0b9
+#include "pt-common.h"
0cb0b9
+PT_define_limit_functions(atari)
0cb0b9
+
0cb0b9
+static PedDiskOps atari_disk_ops = {
0cb0b9
+	clobber:	        NULL_IF_DISCOVER_ONLY (atari_clobber),
0cb0b9
+	write:			NULL_IF_DISCOVER_ONLY (atari_write),
0cb0b9
+
0cb0b9
+	partition_set_name:	NULL,
0cb0b9
+	partition_get_name:	NULL,
0cb0b9
+
0cb0b9
+        PT_op_function_initializers (atari)
0cb0b9
+};
0cb0b9
+
0cb0b9
+static PedDiskType atari_disk_type = {
0cb0b9
+	next:		NULL,
0cb0b9
+	name:		"atari",
0cb0b9
+	ops:		&atari_disk_ops,
0cb0b9
+	features:	PED_DISK_TYPE_EXTENDED
0cb0b9
+};
0cb0b9
+
0cb0b9
+void
0cb0b9
+ped_disk_atari_init ()
0cb0b9
+{
0cb0b9
+	PED_ASSERT (sizeof (AtariRawPartition) == 12);
0cb0b9
+	PED_ASSERT (sizeof (AtariRawTable) == 512);
0cb0b9
+	/* GNU Libc doesn't support NULL instead of the locale name */
0cb0b9
+	PED_ASSERT ((atr_c_locale = newlocale(LC_ALL_MASK, "C", NULL)) != NULL);
0cb0b9
+
0cb0b9
+	ped_disk_type_register (&atari_disk_type);
0cb0b9
+}
0cb0b9
+
0cb0b9
+void
0cb0b9
+ped_disk_atari_done ()
0cb0b9
+{
0cb0b9
+	ped_disk_type_unregister (&atari_disk_type);
0cb0b9
+	freelocale(atr_c_locale);
0cb0b9
+}
0cb0b9
diff --git a/libparted/labels/pt-limit.gperf b/libparted/labels/pt-limit.gperf
0cb0b9
index 3d764ae..d5a580d 100644
0cb0b9
--- a/libparted/labels/pt-limit.gperf
0cb0b9
+++ b/libparted/labels/pt-limit.gperf
0cb0b9
@@ -25,3 +25,4 @@ pc98,UINT32_MAX,UINT32_MAX
0cb0b9
 #
0cb0b9
 # FIXME: not verified.  looks like these are cylinder aligned, too
0cb0b9
 amiga,UINT32_MAX,UINT32_MAX
0cb0b9
+atari,INT32_MAX,INT32_MAX
0cb0b9
diff --git a/libparted/libparted.c b/libparted/libparted.c
0cb0b9
index d5cbb3a..d855d0e 100644
0cb0b9
--- a/libparted/libparted.c
0cb0b9
+++ b/libparted/libparted.c
0cb0b9
@@ -75,6 +75,7 @@ extern void ped_disk_pc98_init ();
0cb0b9
 extern void ped_disk_sun_init ();
0cb0b9
 extern void ped_disk_amiga_init ();
0cb0b9
 extern void ped_disk_dasd_init ();
0cb0b9
+extern void ped_disk_atari_init ();
0cb0b9
 
0cb0b9
 static void
0cb0b9
 init_disk_types ()
0cb0b9
@@ -96,6 +97,7 @@ init_disk_types ()
0cb0b9
 	ped_disk_bsd_init ();
0cb0b9
 	ped_disk_amiga_init ();
0cb0b9
 	ped_disk_aix_init ();
0cb0b9
+	ped_disk_atari_init ();
0cb0b9
 }
0cb0b9
 
0cb0b9
 extern void ped_file_system_amiga_init (void);
0cb0b9
@@ -139,6 +141,7 @@ extern void ped_disk_pc98_done ();
0cb0b9
 extern void ped_disk_sun_done ();
0cb0b9
 extern void ped_disk_amiga_done ();
0cb0b9
 extern void ped_disk_dasd_done ();
0cb0b9
+extern void ped_disk_atari_done ();
0cb0b9
 
0cb0b9
 static void
0cb0b9
 done_disk_types ()
0cb0b9
@@ -158,6 +161,7 @@ done_disk_types ()
0cb0b9
 	ped_disk_bsd_done ();
0cb0b9
 	ped_disk_amiga_done ();
0cb0b9
 	ped_disk_aix_done ();
0cb0b9
+	ped_disk_atari_done ();
0cb0b9
 }
0cb0b9
 
0cb0b9
 static void _init() __attribute__ ((constructor));
0cb0b9
diff --git a/libparted/tests/common.c b/libparted/tests/common.c
0cb0b9
index 9115686..2be0e3a 100644
0cb0b9
--- a/libparted/tests/common.c
0cb0b9
+++ b/libparted/tests/common.c
0cb0b9
@@ -83,6 +83,9 @@ _implemented_disk_label (const char *label)
0cb0b9
         /* FIXME: these have minor problems, so skip them, temporarily.  */
0cb0b9
         if (STREQ (label, "amiga")) return 0;
0cb0b9
 
0cb0b9
+        if (STREQ (label, "atari") && get_sector_size() != 512)
0cb0b9
+                return 0;
0cb0b9
+
0cb0b9
         /* Not implemented yet */
0cb0b9
         if (STREQ (label, "aix")) return 0;
0cb0b9
         if (STREQ (label, "pc98")) return 0;
0cb0b9
diff --git a/po/POTFILES.in b/po/POTFILES.in
0cb0b9
index a8e5994..7a196f7 100644
0cb0b9
--- a/po/POTFILES.in
0cb0b9
+++ b/po/POTFILES.in
0cb0b9
@@ -27,6 +27,7 @@ libparted/debug.c
0cb0b9
 libparted/disk.c
0cb0b9
 libparted/exception.c
0cb0b9
 libparted/labels/aix.c
0cb0b9
+libparted/labels/atari.c
0cb0b9
 libparted/labels/bsd.c
0cb0b9
 libparted/labels/dasd.c
0cb0b9
 libparted/labels/dos.c
0cb0b9
diff --git a/tests/t3310-flags.sh b/tests/t3310-flags.sh
0cb0b9
index 10ac50d..2da72d0 100644
0cb0b9
--- a/tests/t3310-flags.sh
0cb0b9
+++ b/tests/t3310-flags.sh
0cb0b9
@@ -25,7 +25,7 @@ extract_flags()
0cb0b9
   perl -nle '/^[^:]*:4096s:6143s:2048s::[^:]*:(.+);$/ and print $1' "$@"
0cb0b9
 }
0cb0b9
 
0cb0b9
-for table_type in aix amiga bsd dvh gpt mac msdos pc98 sun loop; do
0cb0b9
+for table_type in aix amiga atari bsd dvh gpt mac msdos pc98 sun loop; do
0cb0b9
   ptn_num=1
0cb0b9
 
0cb0b9
   case $table_type in
0cb0b9
@@ -35,6 +35,10 @@ for table_type in aix amiga bsd dvh gpt mac msdos pc98 sun loop; do
0cb0b9
            ;;
0cb0b9
     amiga) primary_or_name='PTNNAME'
0cb0b9
            ;;
0cb0b9
+    atari)  primary_or_name='primary'
0cb0b9
+           # atari only supports 512b sectors
0cb0b9
+           [ $ss -ne 512 ] && continue
0cb0b9
+           ;;
0cb0b9
     bsd)   primary_or_name=''
0cb0b9
            ;;
0cb0b9
     dvh)   primary_or_name='primary'
0cb0b9
diff --git a/tests/t9021-maxima.sh b/tests/t9021-maxima.sh
0cb0b9
index 510e6c2..5320a6e 100755
0cb0b9
--- a/tests/t9021-maxima.sh
0cb0b9
+++ b/tests/t9021-maxima.sh
0cb0b9
@@ -37,6 +37,7 @@ max_n_partitions()
0cb0b9
     mac) m=65536;;
0cb0b9
     bsd) m=8;;
0cb0b9
     amiga) m=128;;
0cb0b9
+    atari) m=12;;
0cb0b9
     loop) m=1;;
0cb0b9
     pc98) case $ss in 512) m=16;; *) m=64;; esac;;
0cb0b9
     *) warn_ invalid partition table type: $1 1>&2; exit 1;;
0cb0b9
@@ -45,8 +46,9 @@ max_n_partitions()
0cb0b9
 }
0cb0b9
 
0cb0b9
 # FIXME: add aix when/if it's supported again
0cb0b9
-for t in msdos gpt dvh sun mac bsd amiga loop pc98; do
0cb0b9
+for t in msdos gpt dvh sun mac bsd amiga atari loop pc98; do
0cb0b9
     echo $t
0cb0b9
+    [ $t == 'atari' ] && [ $ss != 512 ] && continue
0cb0b9
     rm -f $dev
0cb0b9
     dd if=/dev/zero of=$dev bs=$ss count=1 seek=10000 || { fail=1; continue; }
0cb0b9
     parted -s $dev mklabel $t || { fail=1; continue; }
0cb0b9
@@ -56,6 +58,7 @@ for t in msdos gpt dvh sun mac bsd amiga loop pc98; do
0cb0b9
     max_start=4294967295
0cb0b9
     max_len=4294967295
0cb0b9
     case $t in
0cb0b9
+	atari) max_start=2147483647; max_len=$max_start;;
0cb0b9
 	gpt|loop) max_start=18446744073709551615; max_len=$max_start;;
0cb0b9
 	sun) max_start=549755813760;; # 128 * (2^32-1)
0cb0b9
     esac
0cb0b9
-- 
0cb0b9
2.9.3
0cb0b9