2009-03-06 13:26:48 +01:00
|
|
|
{pkgs, config, ...}:
|
2006-12-13 15:24:33 +01:00
|
|
|
|
2009-08-10 20:41:57 +02:00
|
|
|
with pkgs.lib;
|
2009-03-06 13:26:48 +01:00
|
|
|
|
|
|
|
let
|
|
|
|
|
|
|
|
inherit (pkgs) substituteAll stdenv writeText udev procps;
|
2006-12-13 15:24:33 +01:00
|
|
|
|
2008-07-03 12:45:14 +02:00
|
|
|
cfg = config.services.udev;
|
|
|
|
|
2008-07-02 20:06:34 +02:00
|
|
|
firmwareLoader = substituteAll {
|
|
|
|
src = ./udev-firmware-loader.sh;
|
|
|
|
path = "${stdenv.coreutils}/bin";
|
|
|
|
isExecutable = true;
|
2009-08-10 21:05:20 +02:00
|
|
|
firmwareDirs = cfg.addFirmware;
|
2008-07-02 20:06:34 +02:00
|
|
|
};
|
2009-03-06 13:26:48 +01:00
|
|
|
|
2009-08-10 21:05:20 +02:00
|
|
|
extraUdevRules = pkgs.writeTextFile {
|
|
|
|
name = "extra-udev-rules";
|
|
|
|
text = cfg.extraRules;
|
|
|
|
destination = "/etc/udev/rules.d/10-local.rules";
|
|
|
|
};
|
2009-03-06 13:26:48 +01:00
|
|
|
|
|
|
|
modprobe = config.system.sbin.modprobe;
|
2008-07-02 20:06:34 +02:00
|
|
|
|
2009-08-10 21:05:20 +02:00
|
|
|
nixosRules = ''
|
2008-07-02 20:06:34 +02:00
|
|
|
|
|
|
|
# Miscellaneous devices.
|
|
|
|
KERNEL=="sonypi", MODE="0666"
|
|
|
|
KERNEL=="kvm", MODE="0666"
|
2008-07-23 16:13:27 +02:00
|
|
|
KERNEL=="kqemu", NAME="%k", MODE="0666"
|
2009-04-28 15:17:04 +02:00
|
|
|
KERNEL=="vboxdrv", NAME="vboxdrv", OWNER="root", GROUP="root", MODE="0666"
|
2008-07-02 20:06:34 +02:00
|
|
|
|
2008-07-31 16:13:35 +02:00
|
|
|
# Create symlinks for CD/DVD devices.
|
|
|
|
ACTION=="add", SUBSYSTEM=="block", ENV{ID_CDROM}=="?*", SYMLINK+="cdrom cdrom-%k"
|
|
|
|
ACTION=="add", SUBSYSTEM=="block", ENV{ID_CDROM_CD_RW}=="?*", SYMLINK+="cdrw cdrw-%k"
|
|
|
|
ACTION=="add", SUBSYSTEM=="block", ENV{ID_CDROM_DVD}=="?*", SYMLINK+="dvd dvd-%k"
|
|
|
|
ACTION=="add", SUBSYSTEM=="block", ENV{ID_CDROM_DVD_RW}=="?*", SYMLINK+="dvdrw dvdrw-%k"
|
|
|
|
|
2008-07-02 20:06:34 +02:00
|
|
|
# ALSA sound devices.
|
2008-07-03 12:45:14 +02:00
|
|
|
KERNEL=="controlC[0-9]*", NAME="snd/%k", MODE="${cfg.sndMode}"
|
|
|
|
KERNEL=="hwC[D0-9]*", NAME="snd/%k", MODE="${cfg.sndMode}"
|
|
|
|
KERNEL=="pcmC[D0-9cp]*", NAME="snd/%k", MODE="${cfg.sndMode}"
|
|
|
|
KERNEL=="midiC[D0-9]*", NAME="snd/%k", MODE="${cfg.sndMode}"
|
|
|
|
KERNEL=="timer", NAME="snd/%k", MODE="${cfg.sndMode}"
|
|
|
|
KERNEL=="seq", NAME="snd/%k", MODE="${cfg.sndMode}"
|
2008-07-02 20:06:34 +02:00
|
|
|
|
|
|
|
# Firmware loading.
|
|
|
|
SUBSYSTEM=="firmware", ACTION=="add", RUN+="${firmwareLoader}"
|
|
|
|
|
|
|
|
'';
|
|
|
|
|
2007-03-04 00:20:08 +01:00
|
|
|
# Perform substitutions in all udev rules files.
|
|
|
|
udevRules = stdenv.mkDerivation {
|
|
|
|
name = "udev-rules";
|
2008-07-02 20:06:34 +02:00
|
|
|
#src = cleanSource ./udev-rules;
|
|
|
|
buildCommand = ''
|
2007-03-04 00:20:08 +01:00
|
|
|
ensureDir $out
|
2007-07-22 04:07:02 +02:00
|
|
|
shopt -s nullglob
|
2009-08-10 21:05:20 +02:00
|
|
|
|
|
|
|
# Use all the default udev rules.
|
2008-07-31 16:13:35 +02:00
|
|
|
cp ${udev}/*/udev/rules.d/*.rules $out/
|
2009-08-10 21:05:20 +02:00
|
|
|
|
|
|
|
# If auto-configuration is disabled, then remove
|
|
|
|
# udev's 80-drivers.rules file, which contains rules for
|
|
|
|
# automatically calling modprobe.
|
2008-07-03 20:54:16 +02:00
|
|
|
${if config.boot.hardwareScan then
|
|
|
|
''
|
|
|
|
substituteInPlace $out/80-drivers.rules \
|
|
|
|
--replace /sbin/modprobe ${modprobe}/sbin/modprobe
|
2008-07-23 16:13:27 +02:00
|
|
|
''
|
|
|
|
else
|
|
|
|
''
|
|
|
|
rm $out/80-drivers.rules
|
|
|
|
''
|
2008-07-03 20:54:16 +02:00
|
|
|
}
|
2009-08-10 21:05:20 +02:00
|
|
|
|
|
|
|
# Add the udev rules from other packages.
|
|
|
|
for i in ${toString cfg.packages}; do
|
2008-07-31 16:13:35 +02:00
|
|
|
for j in $i/*/udev/rules.d/*; do
|
2007-07-22 04:07:02 +02:00
|
|
|
ln -s $j $out/$(basename $j)
|
|
|
|
done
|
|
|
|
done
|
2008-07-02 20:06:34 +02:00
|
|
|
''; # */
|
2007-03-04 00:20:08 +01:00
|
|
|
};
|
|
|
|
|
2008-08-09 00:44:45 +02:00
|
|
|
# The udev configuration file.
|
2008-02-04 11:52:58 +01:00
|
|
|
conf = writeText "udev.conf" ''
|
|
|
|
udev_rules="${udevRules}"
|
|
|
|
#udev_log="debug"
|
|
|
|
'';
|
2006-12-13 15:24:33 +01:00
|
|
|
|
2007-04-02 17:00:31 +02:00
|
|
|
# Dummy file indicating whether we've run udevtrigger/udevsettle.
|
|
|
|
# Since that *recreates* all device nodes with default permissions,
|
|
|
|
# it's not nice to do that when a user is logged in (it messes up
|
|
|
|
# the permissions set by pam_devperm).
|
|
|
|
# !!! Actually, this makes the udev configuration less declarative;
|
|
|
|
# changes may not take effect until the user reboots. We should
|
|
|
|
# find a better way to preserve the permissions of logged-in users.
|
|
|
|
devicesCreated = "/var/run/devices-created";
|
|
|
|
|
2006-12-13 15:24:33 +01:00
|
|
|
in
|
2006-12-13 13:17:38 +01:00
|
|
|
|
|
|
|
{
|
2008-06-20 18:09:48 +02:00
|
|
|
|
2009-08-10 20:41:57 +02:00
|
|
|
###### interface
|
|
|
|
|
|
|
|
options = {
|
2008-06-20 18:09:48 +02:00
|
|
|
|
2009-08-10 20:41:57 +02:00
|
|
|
boot.hardwareScan = mkOption {
|
|
|
|
default = true;
|
|
|
|
description = ''
|
|
|
|
Whether to try to load kernel modules for all detected hardware.
|
|
|
|
Usually this does a good job of providing you with the modules
|
|
|
|
you need, but sometimes it can crash the system or cause other
|
|
|
|
nasty effects. If the hardware scan is turned on, it can be
|
|
|
|
disabled at boot time by adding the <literal>safemode</literal>
|
|
|
|
parameter to the kernel command line.
|
|
|
|
'';
|
|
|
|
};
|
|
|
|
|
|
|
|
services.udev = {
|
|
|
|
|
|
|
|
addFirmware = mkOption {
|
|
|
|
default = [];
|
|
|
|
example = ["/mnt/big-storage/firmware/"];
|
2009-08-10 21:05:20 +02:00
|
|
|
merge = mergeListOption;
|
2009-08-10 20:41:57 +02:00
|
|
|
description = ''
|
|
|
|
To specify firmware that is not too spread to ensure
|
|
|
|
a package, or have an interactive process of extraction
|
|
|
|
and cannot be redistributed.
|
|
|
|
'';
|
|
|
|
};
|
|
|
|
|
2009-08-10 21:05:20 +02:00
|
|
|
packages = mkOption {
|
2009-08-10 20:41:57 +02:00
|
|
|
default = [];
|
2009-08-10 21:05:20 +02:00
|
|
|
merge = mergeListOption;
|
2009-08-10 20:41:57 +02:00
|
|
|
description = ''
|
|
|
|
List of packages containing <command>udev</command> rules.
|
|
|
|
All files found in
|
2009-08-10 21:05:20 +02:00
|
|
|
<filename><replaceable>pkg</replaceable>/etc/udev/rules.d</filename> and
|
|
|
|
<filename><replaceable>pkg</replaceable>/lib/udev/rules.d</filename>
|
2009-08-10 20:41:57 +02:00
|
|
|
will be included.
|
|
|
|
'';
|
|
|
|
};
|
2009-03-06 13:26:48 +01:00
|
|
|
|
2009-08-10 20:41:57 +02:00
|
|
|
extraRules = mkOption {
|
|
|
|
default = "";
|
|
|
|
example = ''
|
|
|
|
KERNEL=="eth*", ATTR{address}=="00:1D:60:B9:6D:4F", NAME="my_fast_network_card"
|
|
|
|
'';
|
2009-08-10 21:05:20 +02:00
|
|
|
merge = mergeStringOption;
|
2009-08-10 20:41:57 +02:00
|
|
|
description = ''
|
|
|
|
Additional <command>udev</command> rules. They'll be written
|
|
|
|
into file <filename>10-local.rules</filename>. Thus they are
|
|
|
|
read before all other rules.
|
|
|
|
'';
|
|
|
|
};
|
2009-03-06 13:26:48 +01:00
|
|
|
|
2009-08-10 20:41:57 +02:00
|
|
|
sndMode = mkOption {
|
|
|
|
default = "0600";
|
|
|
|
example = "0666";
|
|
|
|
description = ''
|
|
|
|
Permissions for sound devices, in case you have multiple
|
|
|
|
logged in users or if the devices belong to root for some
|
|
|
|
reason.
|
|
|
|
'';
|
|
|
|
};
|
|
|
|
|
|
|
|
};
|
|
|
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
###### implementation
|
|
|
|
|
|
|
|
config = {
|
|
|
|
|
2009-08-10 21:05:20 +02:00
|
|
|
services.udev.extraRules = nixosRules;
|
|
|
|
|
|
|
|
services.udev.packages = [extraUdevRules];
|
|
|
|
|
2009-08-10 20:41:57 +02:00
|
|
|
jobs = singleton
|
|
|
|
{ name = "udev";
|
|
|
|
|
|
|
|
startOn = "startup";
|
|
|
|
stopOn = "shutdown";
|
|
|
|
|
|
|
|
environment = { UDEV_CONFIG_FILE = conf; };
|
|
|
|
|
|
|
|
preStart =
|
|
|
|
''
|
2009-03-06 13:26:48 +01:00
|
|
|
echo "" > /proc/sys/kernel/hotplug
|
|
|
|
|
|
|
|
# Get rid of possible old udev processes.
|
|
|
|
${procps}/bin/pkill -u root "^udevd$" || true
|
|
|
|
|
|
|
|
# Do the loading of additional stage 2 kernel modules.
|
|
|
|
# Maybe this isn't the best place...
|
|
|
|
for i in ${toString config.boot.kernelModules}; do
|
|
|
|
echo "Loading kernel module $i..."
|
|
|
|
${modprobe}/sbin/modprobe $i || true
|
|
|
|
done
|
|
|
|
|
|
|
|
# Start udev.
|
|
|
|
${udev}/sbin/udevd --daemon
|
|
|
|
|
|
|
|
# Let udev create device nodes for all modules that have already
|
|
|
|
# been loaded into the kernel (or for which support is built into
|
|
|
|
# the kernel).
|
|
|
|
if ! test -e ${devicesCreated}; then
|
|
|
|
${udev}/sbin/udevadm trigger
|
|
|
|
${udev}/sbin/udevadm settle # wait for udev to finish
|
|
|
|
touch ${devicesCreated}
|
|
|
|
fi
|
|
|
|
|
|
|
|
# Kill udev, let Upstart restart and monitor it. (This is nasty,
|
|
|
|
# but we have to run `udevadm trigger' first. Maybe we can use
|
|
|
|
# Upstart's `binary' keyword, but it isn't implemented yet.)
|
|
|
|
if ! ${procps}/bin/pkill -u root "^udevd$"; then
|
|
|
|
echo "couldn't stop udevd"
|
|
|
|
fi
|
|
|
|
|
|
|
|
while ${procps}/bin/pgrep -u root "^udevd$"; do
|
|
|
|
sleep 1
|
|
|
|
done
|
|
|
|
|
|
|
|
initctl emit new-devices
|
2009-08-10 20:41:57 +02:00
|
|
|
'';
|
2009-03-06 13:26:48 +01:00
|
|
|
|
2009-08-10 20:41:57 +02:00
|
|
|
exec = "${udev}/sbin/udevd";
|
|
|
|
|
|
|
|
};
|
2009-03-06 13:26:48 +01:00
|
|
|
|
|
|
|
};
|
2009-08-10 20:41:57 +02:00
|
|
|
|
2006-12-13 13:17:38 +01:00
|
|
|
}
|