2014-04-14 16:26:48 +02:00
|
|
|
|
{ config, lib, pkgs, utils, ... }:
|
2009-10-12 18:36:19 +02:00
|
|
|
|
|
2014-04-14 16:26:48 +02:00
|
|
|
|
with lib;
|
2012-10-12 23:32:36 +02:00
|
|
|
|
with utils;
|
2006-12-21 15:22:40 +01:00
|
|
|
|
|
2012-03-15 14:54:23 +01:00
|
|
|
|
let
|
|
|
|
|
|
2012-11-02 18:02:12 +01:00
|
|
|
|
fileSystems = attrValues config.fileSystems;
|
|
|
|
|
|
2013-05-13 23:42:55 +02:00
|
|
|
|
prioOption = prio: optionalString (prio !=null) " pri=${toString prio}";
|
|
|
|
|
|
2014-07-30 13:15:29 +02:00
|
|
|
|
fileSystemOpts = { name, config, ... }: {
|
2012-11-02 18:02:12 +01:00
|
|
|
|
|
|
|
|
|
options = {
|
|
|
|
|
|
|
|
|
|
mountPoint = mkOption {
|
|
|
|
|
example = "/mnt/usb";
|
2013-10-30 11:02:04 +01:00
|
|
|
|
type = types.str;
|
2012-11-02 18:02:12 +01:00
|
|
|
|
description = "Location of the mounted the file system.";
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
device = mkOption {
|
|
|
|
|
default = null;
|
|
|
|
|
example = "/dev/sda";
|
|
|
|
|
type = types.uniq (types.nullOr types.string);
|
|
|
|
|
description = "Location of the device.";
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
label = mkOption {
|
|
|
|
|
default = null;
|
|
|
|
|
example = "root-partition";
|
|
|
|
|
type = types.uniq (types.nullOr types.string);
|
|
|
|
|
description = "Label of the device (if any).";
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
fsType = mkOption {
|
|
|
|
|
default = "auto";
|
|
|
|
|
example = "ext3";
|
2013-10-30 11:02:04 +01:00
|
|
|
|
type = types.str;
|
2012-11-02 18:02:12 +01:00
|
|
|
|
description = "Type of the file system.";
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
options = mkOption {
|
|
|
|
|
default = "defaults,relatime";
|
|
|
|
|
example = "data=journal";
|
2013-10-28 16:14:15 +01:00
|
|
|
|
type = types.commas;
|
2012-11-02 18:02:12 +01:00
|
|
|
|
description = "Options used to mount the file system.";
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
autoFormat = mkOption {
|
|
|
|
|
default = false;
|
|
|
|
|
type = types.bool;
|
|
|
|
|
description = ''
|
|
|
|
|
If the device does not currently contain a filesystem (as
|
|
|
|
|
determined by <command>blkid</command>, then automatically
|
|
|
|
|
format it with the filesystem type specified in
|
|
|
|
|
<option>fsType</option>. Use with caution.
|
|
|
|
|
'';
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
noCheck = mkOption {
|
|
|
|
|
default = false;
|
|
|
|
|
type = types.bool;
|
|
|
|
|
description = "Disable running fsck on this filesystem.";
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
config = {
|
|
|
|
|
mountPoint = mkDefault name;
|
2014-07-30 15:44:47 +02:00
|
|
|
|
device = mkIf (config.fsType == "tmpfs") (mkDefault config.fsType);
|
2012-11-02 18:02:12 +01:00
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
};
|
|
|
|
|
|
2012-03-15 14:54:23 +01:00
|
|
|
|
in
|
2012-10-12 23:32:36 +02:00
|
|
|
|
|
2009-10-12 18:36:19 +02:00
|
|
|
|
{
|
|
|
|
|
|
|
|
|
|
###### interface
|
2009-05-28 01:14:38 +02:00
|
|
|
|
|
|
|
|
|
options = {
|
|
|
|
|
|
2009-06-11 18:03:57 +02:00
|
|
|
|
fileSystems = mkOption {
|
2013-11-18 16:26:39 +01:00
|
|
|
|
default = {};
|
2012-11-02 18:02:12 +01:00
|
|
|
|
example = {
|
|
|
|
|
"/".device = "/dev/hda1";
|
|
|
|
|
"/data" = {
|
2009-05-28 01:14:38 +02:00
|
|
|
|
device = "/dev/hda2";
|
|
|
|
|
fsType = "ext3";
|
|
|
|
|
options = "data=journal";
|
2012-11-02 18:02:12 +01:00
|
|
|
|
};
|
|
|
|
|
"/bigdisk".label = "bigdisk";
|
|
|
|
|
};
|
|
|
|
|
type = types.loaOf types.optionSet;
|
|
|
|
|
options = [ fileSystemOpts ];
|
2012-03-09 15:37:58 +01:00
|
|
|
|
description = ''
|
2009-05-28 01:14:38 +02:00
|
|
|
|
The file systems to be mounted. It must include an entry for
|
2013-10-28 13:36:45 +01:00
|
|
|
|
the root directory (<literal>mountPoint = "/"</literal>). Each
|
2009-05-28 01:14:38 +02:00
|
|
|
|
entry in the list is an attribute set with the following fields:
|
|
|
|
|
<literal>mountPoint</literal>, <literal>device</literal>,
|
|
|
|
|
<literal>fsType</literal> (a file system type recognised by
|
|
|
|
|
<command>mount</command>; defaults to
|
2013-10-28 13:36:45 +01:00
|
|
|
|
<literal>"auto"</literal>), and <literal>options</literal>
|
2009-05-28 01:14:38 +02:00
|
|
|
|
(the mount options passed to <command>mount</command> using the
|
2013-10-28 13:36:45 +01:00
|
|
|
|
<option>-o</option> flag; defaults to <literal>"defaults"</literal>).
|
2009-05-28 01:14:38 +02:00
|
|
|
|
|
|
|
|
|
Instead of specifying <literal>device</literal>, you can also
|
|
|
|
|
specify a volume label (<literal>label</literal>) for file
|
|
|
|
|
systems that support it, such as ext2/ext3 (see <command>mke2fs
|
|
|
|
|
-L</command>).
|
2012-03-09 15:37:58 +01:00
|
|
|
|
'';
|
2009-05-28 01:14:38 +02:00
|
|
|
|
};
|
2011-09-14 20:20:50 +02:00
|
|
|
|
|
2012-03-09 15:37:58 +01:00
|
|
|
|
system.fsPackages = mkOption {
|
2009-05-28 01:14:38 +02:00
|
|
|
|
internal = true;
|
2012-03-09 15:37:58 +01:00
|
|
|
|
default = [ ];
|
|
|
|
|
description = "Packages supplying file system mounters and checkers.";
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
boot.supportedFilesystems = mkOption {
|
|
|
|
|
default = [ ];
|
|
|
|
|
example = [ "btrfs" ];
|
2013-03-14 17:09:21 +01:00
|
|
|
|
type = types.listOf types.string;
|
2012-03-09 15:37:58 +01:00
|
|
|
|
description = "Names of supported filesystem types.";
|
|
|
|
|
};
|
|
|
|
|
|
2009-05-28 01:14:38 +02:00
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
2009-10-12 18:36:19 +02:00
|
|
|
|
###### implementation
|
2009-05-28 01:14:38 +02:00
|
|
|
|
|
2009-10-12 18:36:19 +02:00
|
|
|
|
config = {
|
2009-05-28 01:14:38 +02:00
|
|
|
|
|
2012-11-02 18:02:12 +01:00
|
|
|
|
boot.supportedFilesystems = map (fs: fs.fsType) fileSystems;
|
2012-03-09 15:37:58 +01:00
|
|
|
|
|
2009-10-12 18:36:19 +02:00
|
|
|
|
# Add the mount helpers to the system path so that `mount' can find them.
|
2012-03-09 15:37:58 +01:00
|
|
|
|
system.fsPackages = [ pkgs.dosfstools ];
|
2012-10-12 23:32:36 +02:00
|
|
|
|
|
2015-04-19 21:27:52 +02:00
|
|
|
|
environment.systemPackages = [ pkgs.fuse ] ++ config.system.fsPackages;
|
2011-09-14 20:20:50 +02:00
|
|
|
|
|
2013-02-03 14:12:49 +01:00
|
|
|
|
environment.etc.fstab.text =
|
2014-12-29 02:53:37 +01:00
|
|
|
|
let
|
|
|
|
|
fsToSkipCheck = [ "none" "btrfs" "zfs" "tmpfs" "nfs" ];
|
|
|
|
|
skipCheck = fs: fs.noCheck || fs.device == "none" || builtins.elem fs.fsType fsToSkipCheck;
|
|
|
|
|
in ''
|
2013-02-03 14:12:49 +01:00
|
|
|
|
# This is a generated file. Do not edit!
|
|
|
|
|
|
|
|
|
|
# Filesystems.
|
|
|
|
|
${flip concatMapStrings fileSystems (fs:
|
2014-07-30 13:10:03 +02:00
|
|
|
|
(if fs.device != null then fs.device
|
|
|
|
|
else if fs.label != null then "/dev/disk/by-label/${fs.label}"
|
|
|
|
|
else throw "No device specified for mount point ‘${fs.mountPoint}’.")
|
2013-02-03 14:12:49 +01:00
|
|
|
|
+ " " + fs.mountPoint
|
|
|
|
|
+ " " + fs.fsType
|
|
|
|
|
+ " " + fs.options
|
|
|
|
|
+ " 0"
|
2014-12-29 02:53:37 +01:00
|
|
|
|
+ " " + (if skipCheck fs then "0" else
|
2013-02-03 14:12:49 +01:00
|
|
|
|
if fs.mountPoint == "/" then "1" else "2")
|
|
|
|
|
+ "\n"
|
|
|
|
|
)}
|
|
|
|
|
|
|
|
|
|
# Swap devices.
|
|
|
|
|
${flip concatMapStrings config.swapDevices (sw:
|
2013-05-13 23:42:55 +02:00
|
|
|
|
"${sw.device} none swap${prioOption sw.priority}\n"
|
2013-02-03 14:12:49 +01:00
|
|
|
|
)}
|
|
|
|
|
'';
|
2010-06-04 16:22:11 +02:00
|
|
|
|
|
2012-10-12 21:08:44 +02:00
|
|
|
|
# Provide a target that pulls in all filesystems.
|
2013-01-16 12:33:18 +01:00
|
|
|
|
systemd.targets.fs =
|
2012-10-12 21:08:44 +02:00
|
|
|
|
{ description = "All File Systems";
|
|
|
|
|
wants = [ "local-fs.target" "remote-fs.target" ];
|
|
|
|
|
};
|
|
|
|
|
|
2012-10-12 23:32:36 +02:00
|
|
|
|
# Emit systemd services to format requested filesystems.
|
2013-01-16 12:33:18 +01:00
|
|
|
|
systemd.services =
|
2012-10-12 23:32:36 +02:00
|
|
|
|
let
|
|
|
|
|
|
|
|
|
|
formatDevice = fs:
|
|
|
|
|
let
|
|
|
|
|
mountPoint' = escapeSystemdPath fs.mountPoint;
|
|
|
|
|
device' = escapeSystemdPath fs.device;
|
2014-05-09 14:32:12 +02:00
|
|
|
|
# -F needed to allow bare block device without partitions
|
|
|
|
|
mkfsOpts = optional ((builtins.substring 0 3 fs.fsType) == "ext") "-F";
|
2012-10-12 23:32:36 +02:00
|
|
|
|
in nameValuePair "mkfs-${device'}"
|
|
|
|
|
{ description = "Initialisation of Filesystem ${fs.device}";
|
|
|
|
|
wantedBy = [ "${mountPoint'}.mount" ];
|
2012-10-31 14:49:09 +01:00
|
|
|
|
before = [ "${mountPoint'}.mount" "systemd-fsck@${device'}.service" ];
|
2013-06-25 16:46:23 +02:00
|
|
|
|
requires = [ "${device'}.device" ];
|
2012-10-12 23:32:36 +02:00
|
|
|
|
after = [ "${device'}.device" ];
|
|
|
|
|
path = [ pkgs.utillinux ] ++ config.system.fsPackages;
|
|
|
|
|
script =
|
|
|
|
|
''
|
|
|
|
|
if ! [ -e "${fs.device}" ]; then exit 1; fi
|
|
|
|
|
# FIXME: this is scary. The test could be more robust.
|
2012-05-16 02:03:44 +02:00
|
|
|
|
type=$(blkid -p -s TYPE -o value "${fs.device}" || true)
|
|
|
|
|
if [ -z "$type" ]; then
|
|
|
|
|
echo "creating ${fs.fsType} filesystem on ${fs.device}..."
|
2014-05-09 14:32:12 +02:00
|
|
|
|
mkfs.${fs.fsType} ${concatStringsSep " " mkfsOpts} "${fs.device}"
|
2012-05-16 02:03:44 +02:00
|
|
|
|
fi
|
2012-10-12 23:32:36 +02:00
|
|
|
|
'';
|
|
|
|
|
unitConfig.RequiresMountsFor = [ "${dirOf fs.device}" ];
|
|
|
|
|
unitConfig.DefaultDependencies = false; # needed to prevent a cycle
|
|
|
|
|
serviceConfig.Type = "oneshot";
|
|
|
|
|
};
|
2010-06-07 14:15:55 +02:00
|
|
|
|
|
2012-11-02 18:02:12 +01:00
|
|
|
|
in listToAttrs (map formatDevice (filter (fs: fs.autoFormat) fileSystems));
|
2006-12-21 15:22:40 +01:00
|
|
|
|
|
2009-03-06 13:27:33 +01:00
|
|
|
|
};
|
2009-08-11 23:10:33 +02:00
|
|
|
|
|
2006-12-21 15:22:40 +01:00
|
|
|
|
}
|