Skip to content

Commit 9dabcb4

Browse files
Initial addition of granular bcachefs formatting and multi-disk support
These changes provide advanced usage of bcachefs formatting options. Because Bcachefs provides a one-stop-shop for wiping, partitioning, formatting and mounting with unique features for each step, it has been broken out into it's own type if users wish to go beyond a single disk setup. examples/bcachefs.nix has been updated to showcase this advanced usage and some of the avaliable options. Because all three processes take place via the type object, requiring "disks" attribute set is not longer a hard requirement if a bcachefs type is provided. A check has been added to account for that use case.
1 parent 18d0a98 commit 9dabcb4

File tree

5 files changed

+216
-7
lines changed

5 files changed

+216
-7
lines changed

example/bcachefs.nix

+28
Original file line numberDiff line numberDiff line change
@@ -1,4 +1,5 @@
11
{
2+
# Basic usage
23
disko.devices = {
34
disk = {
45
main = {
@@ -30,5 +31,32 @@
3031
};
3132
};
3233
};
34+
35+
# Advanced Usage
36+
bcachefs = {
37+
pool = {
38+
type = "bcachefs";
39+
devices = {
40+
disk1 = {
41+
type = "disk";
42+
device = "/dev/sda";
43+
label = "fast";
44+
discard = true;
45+
dataAllowed = [ "journal" "btree" ];
46+
};
47+
disk2 = {
48+
type = "disk";
49+
device = "/dev/sdb";
50+
label = "slow";
51+
durability = 2;
52+
dataAllowed = [ "user" ];
53+
};
54+
};
55+
formatOptions = [ "--compression=zstd" ];
56+
mountPoint = "/mnt/pool";
57+
mountOptions = [ "verbose" "degraded" ];
58+
};
3359
};
60+
};
61+
3462
}

flake.lock

+3-3
Some generated files are not rendered by default. Learn more about customizing how changed files appear on GitHub.

lib/default.nix

+11-3
Original file line numberDiff line numberDiff line change
@@ -397,11 +397,11 @@ let
397397
};
398398
};
399399

400-
/* topLevel type of the disko config, takes attrsets of disks, mdadms, zpools, nodevs, and lvm vgs.
400+
/* topLevel type of the disko config, takes attrsets of disks, bcachefs, mdadms, zpools, nodevs, and lvm vgs.
401401
*/
402402
toplevel = lib.types.submodule (cfg:
403403
let
404-
devices = { inherit (cfg.config) disk mdadm zpool lvm_vg nodev; };
404+
devices = { inherit (cfg.config) disk bcachefs mdadm zpool lvm_vg nodev; };
405405
in
406406
{
407407
options = {
@@ -410,6 +410,11 @@ let
410410
default = { };
411411
description = "Block device";
412412
};
413+
bcachefs = lib.mkOption {
414+
type = lib.types.attrsOf diskoLib.types.bcachefs;
415+
default = { };
416+
description = "bcachefs device";
417+
};
413418
mdadm = lib.mkOption {
414419
type = lib.types.attrsOf diskoLib.types.mdadm;
415420
default = { };
@@ -453,7 +458,10 @@ let
453458
'';
454459
default = { pkgs, checked ? false }:
455460
let
456-
throwIfNoDisksDetected = _: v: if devices.disk == { } then throw "No disks defined, did you forget to import your disko config?" else v;
461+
throwIfNoDisksDetected = _: v: if devices.disk == { } && devices.bcachefs == { }
462+
then throw "No disks or bcachefs devices defined, did you forget to import your disko config?"
463+
else v;
464+
457465
destroyDependencies = with pkgs; [
458466
util-linux
459467
e2fsprogs

lib/types/bcachefs.nix

+171
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,171 @@
1+
{ config, options, lib, diskoLib, ... }:
2+
let
3+
deviceSubmodule = lib.types.submodule {
4+
options = {
5+
type = lib.mkOption {
6+
type = lib.types.enum [ "disk" ];
7+
default = "disk";
8+
description = "Device type (must be disk for bcachefs)";
9+
};
10+
device = lib.mkOption {
11+
type = lib.types.str;
12+
description = "Device path";
13+
};
14+
label = lib.mkOption {
15+
type = lib.types.nullOr lib.types.str;
16+
default = null;
17+
description = "Device label";
18+
};
19+
discard = lib.mkOption {
20+
type = lib.types.bool;
21+
default = false;
22+
description = "Enable TRIM/discard";
23+
};
24+
fsSize = lib.mkOption {
25+
type = lib.types.nullOr lib.types.str;
26+
default = null;
27+
description = "Filesystem size";
28+
};
29+
bucketSize = lib.mkOption {
30+
type = lib.types.nullOr lib.types.str;
31+
default = null;
32+
description = "Bucket size";
33+
};
34+
durability = lib.mkOption {
35+
type = lib.types.nullOr lib.types.int;
36+
default = null;
37+
description = "Replication factor";
38+
};
39+
dataAllowed = lib.mkOption {
40+
type = lib.types.listOf (lib.types.enum [ "journal" "btree" "user" ]);
41+
default = [];
42+
description = "Allowed data types";
43+
};
44+
};
45+
};
46+
firstDevice = (lib.head (lib.attrValues config.devices)).device;
47+
in
48+
{
49+
options = {
50+
type = lib.mkOption {
51+
type = lib.types.enum [ "bcachefs" ];
52+
default = "bcachefs";
53+
internal = true;
54+
description = "Type";
55+
};
56+
formatOptions = lib.mkOption {
57+
type = lib.types.listOf lib.types.str;
58+
default = [];
59+
description = "Additional options for bcachefs format";
60+
};
61+
devices = lib.mkOption {
62+
type = lib.types.attrsOf deviceSubmodule;
63+
default = {};
64+
description = "Named set of devices for the bcachefs filesystem";
65+
example = lib.literalExpression ''
66+
{
67+
disk1 = {
68+
type = "disk";
69+
device = "/dev/sda";
70+
label = "main";
71+
discard = true;
72+
};
73+
disk2 = {
74+
type = "disk";
75+
device = "/dev/sdb";
76+
label = "backup";
77+
durability = 2;
78+
};
79+
}
80+
'';
81+
};
82+
mountPoint = lib.mkOption {
83+
type = lib.types.str;
84+
default = "";
85+
description = "Allow mount directly rather via content because bcachefs provides one-stop-shop";
86+
example = "/mnt/pool";
87+
};
88+
mountOptions = lib.mkOption {
89+
type = lib.types.listOf lib.types.str;
90+
default = [ "defaults" ];
91+
description = "Options to pass to mount";
92+
};
93+
_meta = lib.mkOption {
94+
internal = true;
95+
readOnly = true;
96+
type = diskoLib.jsonType;
97+
default = {};
98+
};
99+
_create = diskoLib.mkCreateOption {
100+
inherit config options;
101+
default = let
102+
# Build device-specific arguments for each device
103+
deviceArgs = lib.concatMap (dev:
104+
(lib.optional (dev.label != null) "--label=${dev.label}")
105+
++ (lib.optional dev.discard "--discard")
106+
++ (lib.optional (dev.fsSize != null) "--fs_size=${dev.fsSize}")
107+
++ (lib.optional (dev.bucketSize != null) "--bucket=${dev.bucketSize}")
108+
++ (lib.optional (dev.durability != null) "--durability=${toString dev.durability}")
109+
++ (lib.optional (dev.dataAllowed != []) "--data_allowed=${lib.concatStringsSep "," dev.dataAllowed}")
110+
++ [ dev.device ]
111+
) (lib.attrValues config.devices);
112+
allArgs = config.formatOptions ++ deviceArgs;
113+
in ''
114+
# Check if any of the devices are already formatted with bcachefs
115+
needs_format=0
116+
for device in ${lib.concatMapStringsSep " " (dev: lib.escapeShellArg dev.device) (lib.attrValues config.devices)}; do
117+
if ! bcachefs show-super "$device" >/dev/null 2>&1; then
118+
needs_format=1
119+
break
120+
fi
121+
done
122+
123+
if [ "$needs_format" -eq 1 ]; then
124+
bcachefs format --force ${lib.concatStringsSep " " allArgs}
125+
udevadm trigger --subsystem-match=block
126+
udevadm settle
127+
fi
128+
'';
129+
};
130+
_mount = diskoLib.mkMountOption {
131+
inherit config options;
132+
default = {
133+
fs = lib.optionalAttrs (config.mountpoint != null) {
134+
${config.mountpoint} = ''
135+
if ! findmnt "${config.mountpoint}" > /dev/null 2>&1; then
136+
UUID=$(bcachefs show-super ${lib.escapeShellArg firstDevice} | grep Ext | awk '{print $3}')
137+
mount -t bcachefs UUID="$UUID" "${config.mountpoint}" \
138+
${lib.concatMapStringsSep " " (opt: "-o ${opt}") config.mountOptions} \
139+
-o X-mount.mkdir
140+
fi
141+
'';
142+
};
143+
};
144+
};
145+
_unmount = diskoLib.mkUnmountOption {
146+
inherit config options;
147+
default = {
148+
fs = lib.optionalAttrs (config.mountpoint != null) {
149+
${config.mountpoint} = ''
150+
if findmnt "${config.mountpoint}" > /dev/null 2>&1; then
151+
umount "${config.mountpoint}"
152+
fi
153+
'';
154+
};
155+
};
156+
};
157+
_config = lib.mkOption {
158+
internal = true;
159+
readOnly = true;
160+
default = [
161+
{ boot.supportedFilesystems = [ "bcachefs" ]; }
162+
];
163+
};
164+
_pkgs = lib.mkOption {
165+
internal = true;
166+
readOnly = true;
167+
type = lib.types.functionTo (lib.types.listOf lib.types.package);
168+
default = pkgs: [ pkgs.bcachefs-tools ];
169+
};
170+
};
171+
}

module.nix

+3-1
Original file line numberDiff line numberDiff line change
@@ -231,7 +231,9 @@ in
231231

232232
system.build = (cfg.devices._scripts { inherit pkgs; checked = cfg.checkScripts; }) // (
233233
let
234-
throwIfNoDisksDetected = _: v: if cfg.devices.disk == { } then throw "No disks defined, did you forget to import your disko config?" else v;
234+
throwIfNoDisksDetected = _: v: if cfg.devices.disk == { } && cfg.devices.bcachefs == { }
235+
then throw "No disks or bcachefs devices defined, did you forget to import your disko config?"
236+
else v;
235237
in
236238
lib.mapAttrs throwIfNoDisksDetected {
237239
# we keep these old outputs for compatibility

0 commit comments

Comments
 (0)