2008-02-08 15:59:15 +00:00
|
|
|
#! @perl@/bin/perl -w
|
2008-01-30 14:16:38 +00:00
|
|
|
|
|
|
|
use File::Spec;
|
|
|
|
use File::Basename;
|
|
|
|
|
|
|
|
|
2009-08-04 09:50:02 +01:00
|
|
|
my @attrs = ();
|
2008-01-30 14:16:38 +00:00
|
|
|
my @kernelModules = ();
|
|
|
|
my @initrdKernelModules = ();
|
|
|
|
|
|
|
|
|
2008-02-08 15:59:15 +00:00
|
|
|
sub debug {
|
|
|
|
return unless defined $ENV{"DEBUG"};
|
|
|
|
print STDERR @_;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2008-01-30 14:16:38 +00:00
|
|
|
# Read a file, returning undef if the file cannot be opened.
|
|
|
|
sub readFile {
|
|
|
|
my $filename = shift;
|
|
|
|
my $res;
|
|
|
|
if (open FILE, "<$filename") {
|
|
|
|
my $prev = $/;
|
|
|
|
undef $/;
|
|
|
|
$res = <FILE>;
|
|
|
|
$/ = $prev;
|
|
|
|
close FILE;
|
|
|
|
chomp $res;
|
|
|
|
}
|
|
|
|
return $res;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
my $cpuinfo = readFile "/proc/cpuinfo";
|
|
|
|
|
|
|
|
|
|
|
|
sub hasCPUFeature {
|
|
|
|
my $feature = shift;
|
|
|
|
return $cpuinfo =~ /^flags\s*:.* $feature( |$)/m;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
# Detect the number of CPU cores.
|
|
|
|
my $cpus = scalar (grep {/^processor\s*:/} (split '\n', $cpuinfo));
|
|
|
|
|
|
|
|
|
|
|
|
# CPU frequency scaling. Not sure about this test.
|
|
|
|
push @kernelModules, "acpi-cpufreq" if hasCPUFeature "acpi";
|
|
|
|
|
|
|
|
|
|
|
|
# Virtualization support?
|
|
|
|
push @kernelModules, "kvm-intel" if hasCPUFeature "vmx";
|
|
|
|
push @kernelModules, "kvm-amd" if hasCPUFeature "svm";
|
|
|
|
|
|
|
|
|
|
|
|
# Look at the PCI devices and add necessary modules. Note that most
|
|
|
|
# modules are auto-detected so we don't need to list them here.
|
|
|
|
# However, some are needed in the initrd to boot the system.
|
|
|
|
|
2008-01-30 14:50:25 +00:00
|
|
|
my $videoDriver = "vesa";
|
2008-01-30 14:32:02 +00:00
|
|
|
|
2008-01-30 14:16:38 +00:00
|
|
|
sub pciCheck {
|
|
|
|
my $path = shift;
|
|
|
|
my $vendor = readFile "$path/vendor";
|
|
|
|
my $device = readFile "$path/device";
|
|
|
|
my $class = readFile "$path/class";
|
|
|
|
|
|
|
|
my $module;
|
|
|
|
if (-e "$path/driver/module") {
|
|
|
|
$module = basename `readlink -f $path/driver/module`;
|
|
|
|
chomp $module;
|
|
|
|
}
|
|
|
|
|
2008-02-08 15:59:15 +00:00
|
|
|
debug "$path: $vendor $device $class";
|
|
|
|
debug " $module" if defined $module;
|
|
|
|
debug "\n";
|
2008-01-30 14:16:38 +00:00
|
|
|
|
|
|
|
if (defined $module) {
|
|
|
|
# See the bottom of http://pciids.sourceforge.net/pci.ids for
|
|
|
|
# device classes.
|
|
|
|
if (# Mass-storage controller. Definitely important.
|
|
|
|
$class =~ /^0x01/ ||
|
|
|
|
|
|
|
|
# Firewire controller. A disk might be attached.
|
|
|
|
$class =~ /^0x0c00/ ||
|
|
|
|
|
|
|
|
# USB controller. Needed if we want to use the
|
|
|
|
# keyboard when things go wrong in the initrd.
|
|
|
|
$class =~ /^0x0c03/
|
|
|
|
)
|
|
|
|
{
|
|
|
|
push @initrdKernelModules, $module;
|
|
|
|
}
|
2008-01-30 14:32:02 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
# Can't rely on $module here, since the module may not be loaded
|
|
|
|
# due to missing firmware. Ideally we would check modules.pcimap
|
|
|
|
# here.
|
2009-08-04 09:50:02 +01:00
|
|
|
push @attrs, "networking.enableIntel2200BGFirmware = true;" if
|
2009-01-25 15:48:59 +00:00
|
|
|
$vendor eq "0x8086" &&
|
2008-01-30 14:32:02 +00:00
|
|
|
($device eq "0x1043" || $device eq "0x104f" || $device eq "0x4220" ||
|
|
|
|
$device eq "0x4221" || $device eq "0x4223" || $device eq "0x4224");
|
2008-01-30 14:50:25 +00:00
|
|
|
|
2009-08-04 09:50:02 +01:00
|
|
|
push @attrs, "networking.enableIntel3945ABGFirmware = true;" if
|
2009-01-25 15:48:59 +00:00
|
|
|
$vendor eq "0x8086" &&
|
2008-02-04 10:39:06 +00:00
|
|
|
($device eq "0x4229" || $device eq "0x4230" ||
|
|
|
|
$device eq "0x4222" || $device eq "0x4227");
|
|
|
|
|
2008-01-30 14:50:25 +00:00
|
|
|
# Hm, can we extract the PCI ids supported by X drivers somehow?
|
|
|
|
# cf. http://www.calel.org/pci-devices/xorg-device-list.html
|
2009-06-17 11:41:17 +01:00
|
|
|
$videoDriver = "intel" if $vendor eq "0x8086" &&
|
2008-01-30 14:50:25 +00:00
|
|
|
($device eq "0x1132" ||
|
|
|
|
$device eq "0x2572" ||
|
|
|
|
$device eq "0x2592" ||
|
|
|
|
$device eq "0x2772" ||
|
|
|
|
$device eq "0x2776" ||
|
|
|
|
$device eq "0x2782" ||
|
|
|
|
$device eq "0x2792" ||
|
|
|
|
$device eq "0x2792" ||
|
|
|
|
$device eq "0x27a2" ||
|
|
|
|
$device eq "0x27a6" ||
|
2008-01-30 14:57:13 +00:00
|
|
|
$device eq "0x29a2" ||
|
2008-01-30 14:50:25 +00:00
|
|
|
$device eq "0x3582" ||
|
|
|
|
$device eq "0x7121" ||
|
|
|
|
$device eq "0x7123" ||
|
|
|
|
$device eq "0x7125" ||
|
|
|
|
$device eq "0x7128"
|
|
|
|
);
|
2008-01-30 14:53:06 +00:00
|
|
|
|
|
|
|
# Assume that all NVIDIA cards are supported by the NVIDIA driver.
|
|
|
|
# There may be exceptions (e.g. old cards).
|
|
|
|
$videoDriver = "nvidia" if $vendor eq "0x10de" && $class =~ /^0x03/;
|
2008-01-30 14:16:38 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
foreach my $path (glob "/sys/bus/pci/devices/*") {
|
|
|
|
pciCheck $path;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
# Idem for USB devices.
|
|
|
|
|
|
|
|
sub usbCheck {
|
|
|
|
my $path = shift;
|
|
|
|
my $class = readFile "$path/bInterfaceClass";
|
|
|
|
my $subclass = readFile "$path/bInterfaceSubClass";
|
|
|
|
my $protocol = readFile "$path/bInterfaceProtocol";
|
|
|
|
|
|
|
|
my $module;
|
|
|
|
if (-e "$path/driver/module") {
|
|
|
|
$module = basename `readlink -f $path/driver/module`;
|
|
|
|
chomp $module;
|
|
|
|
}
|
|
|
|
|
2008-02-08 15:59:15 +00:00
|
|
|
debug "$path: $class $subclass $protocol";
|
|
|
|
debug " $module" if defined $module;
|
|
|
|
debug "\n";
|
2008-01-30 14:16:38 +00:00
|
|
|
|
|
|
|
if (defined $module) {
|
|
|
|
if (# Mass-storage controller. Definitely important.
|
|
|
|
$class eq "08" ||
|
|
|
|
|
|
|
|
# Keyboard. Needed if we want to use the
|
|
|
|
# keyboard when things go wrong in the initrd.
|
|
|
|
($class eq "03" && $protocol eq "01")
|
|
|
|
)
|
|
|
|
{
|
|
|
|
push @initrdKernelModules, $module;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
foreach my $path (glob "/sys/bus/usb/devices/*") {
|
|
|
|
if (-e "$path/bInterfaceClass") {
|
|
|
|
usbCheck $path;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
# Generate the configuration file.
|
|
|
|
|
|
|
|
sub removeDups {
|
|
|
|
my %seen;
|
|
|
|
my @res = ();
|
|
|
|
foreach my $s (@_) {
|
|
|
|
if (!defined $seen{$s}) {
|
|
|
|
$seen{$s} = "";
|
|
|
|
push @res, $s;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return @res;
|
|
|
|
}
|
|
|
|
|
|
|
|
sub toNixExpr {
|
|
|
|
my $res = "";
|
|
|
|
foreach my $s (@_) {
|
|
|
|
$res .= " \"$s\"";
|
|
|
|
}
|
|
|
|
return $res;
|
|
|
|
}
|
|
|
|
|
2009-01-25 15:48:59 +00:00
|
|
|
sub multiLineList {
|
|
|
|
my $indent = shift;
|
|
|
|
my $res = "";
|
|
|
|
foreach my $s (@_) {
|
2009-08-04 09:50:02 +01:00
|
|
|
$res .= "\n$indent$s";
|
2009-01-25 15:48:59 +00:00
|
|
|
}
|
2009-06-17 11:41:17 +01:00
|
|
|
$res .= "\n$indent";
|
2009-01-25 15:48:59 +00:00
|
|
|
return $res;
|
|
|
|
}
|
|
|
|
|
2008-01-30 14:16:38 +00:00
|
|
|
my $initrdKernelModules = toNixExpr(removeDups @initrdKernelModules);
|
|
|
|
my $kernelModules = toNixExpr(removeDups @kernelModules);
|
2009-08-04 09:50:02 +01:00
|
|
|
my $attrs = multiLineList(" ", removeDups @attrs);
|
2008-01-30 14:16:38 +00:00
|
|
|
|
|
|
|
print <<EOF ;
|
2009-06-17 11:41:17 +01:00
|
|
|
# This is a generated file. Do not modify!
|
|
|
|
# Make changes to /etc/nixos/configuration.nix instead.
|
2008-01-30 14:16:38 +00:00
|
|
|
{
|
2009-12-15 14:05:01 +00:00
|
|
|
boot.initrd.kernelModules = [ $initrdKernelModules ];
|
2009-06-17 11:41:17 +01:00
|
|
|
boot.kernelModules = [ $kernelModules ];
|
2008-01-30 14:16:38 +00:00
|
|
|
|
2009-06-17 11:41:17 +01:00
|
|
|
nix.maxJobs = $cpus;
|
2008-01-30 14:32:02 +00:00
|
|
|
|
2009-06-17 11:41:17 +01:00
|
|
|
services.xserver.videoDriver = "$videoDriver";
|
2009-08-04 09:50:02 +01:00
|
|
|
$attrs
|
2008-01-30 14:16:38 +00:00
|
|
|
}
|
|
|
|
EOF
|