#!/usr/bin/env perl # Current Version : 1.1 # Copyright (C) 2008 - 2011 # Arnaud Quette # dloic (loic.dardant AT gmail DOT com) # # Based on the usbdevice.pl script, made for the Ubuntu Media Center # for the final use of the LIRC project. # # This program is free software; you can redistribute it and/or modify # it under the terms of the GNU General Public License as published by # the Free Software Foundation; either version 2 of the License, or # (at your option) any later version. # # This program is distributed in the hope that it will be useful, # but WITHOUT ANY WARRANTY; without even the implied warranty of # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the # GNU General Public License for more details. # # You should have received a copy of the GNU General Public License # along with this program; if not, write to the Free Software # Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA use File::Find; use strict; # path to scan for USB_DEVICE pattern my $scanPath="../drivers"; # HAL output file my $outputHAL="../scripts/hal/ups-nut-device.fdi.in"; # Hotplug output file my $outputHotplug="../scripts/hotplug/libhid.usermap"; # udev output file my $outputUdev="../scripts/udev/nut-usbups.rules.in"; # UPower output file my $outputUPower="../scripts/upower/95-upower-hid.rules"; # tmp output, to allow generating the ENV{UPOWER_VENDOR} header list my $tmpOutputUPower; # mfr header flag my $upowerMfrHeaderDone = 0; # NUT device scanner - C header my $outputDevScanner = "./nut-scanner/nutscan-usb.h"; my $GPL_header = "\ * Copyright (C) 2011 - Arnaud Quette \ *\ * This program is free software; you can redistribute it and/or modify\ * it under the terms of the GNU General Public License as published by\ * the Free Software Foundation; either version 2 of the License, or\ * (at your option) any later version.\ *\ * This program is distributed in the hope that it will be useful,\ * but WITHOUT ANY WARRANTY; without even the implied warranty of\ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the\ * GNU General Public License for more details.\ *\ * You should have received a copy of the GNU General Public License\ * along with this program; if not, write to the Free Software\ * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA"; # array of products indexed by vendorID my %vendor; # contain for each vendor, its name (and...) my %vendorName; ################# MAIN ################# find(\&find_usbdevs,$scanPath); &gen_usb_files; ################# SUB METHOD ################# sub gen_usb_files { # HAL file header open my $outHAL, ">$outputHAL" || die "error $outputHAL : $!"; print $outHAL ' '."\n"; print $outHAL ''."\n"; print $outHAL ' '."\n"; print $outHAL ' '."\n"; # Hotplug file header open my $outHotplug, ">$outputHotplug" || die "error $outputHotplug : $!"; print $outHotplug '# This file is generated and installed by the Network UPS Tools package.'."\n"; print $outHotplug "#\n"; print $outHotplug '# Sample entry (replace 0xVVVV and 0xPPPP with vendor ID and product ID respectively) :'."\n"; print $outHotplug '# libhidups 0x0003 0xVVVV 0xPPPP 0x0000 0x0000 0x00 0x00'; print $outHotplug ' 0x00 0x00 0x00 0x00 0x00000000'."\n"; print $outHotplug "#\n"; print $outHotplug '# usb module match_flags idVendor idProduct bcdDevice_lo bcdDevice_hi'; print $outHotplug ' bDeviceClass bDeviceSubClass bDeviceProtocol bInterfaceClass bInterfaceSubClass'; print $outHotplug ' bInterfaceProtocol driver_info'."\n"; # Udev file header open my $outUdev, ">$outputUdev" || die "error $outputUdev : $!"; print $outUdev '# This file is generated and installed by the Network UPS Tools package.'."\n\n"; print $outUdev 'ACTION!="add|change", GOTO="nut-usbups_rules_end"'."\n"; print $outUdev 'SUBSYSTEM=="usb_device", GOTO="nut-usbups_rules_real"'."\n"; print $outUdev 'SUBSYSTEM=="usb", GOTO="nut-usbups_rules_real"'."\n"; print $outUdev 'SUBSYSTEM!="usb", GOTO="nut-usbups_rules_end"'."\n\n"; print $outUdev 'LABEL="nut-usbups_rules_real"'."\n"; # UPower file header open my $outputUPower, ">$outputUPower" || die "error $outputUPower : $!"; print $outputUPower '##############################################################################################################'."\n"; print $outputUPower '# Uninterruptible Power Supplies with USB HID interfaces'."\n#\n"; print $outputUPower '# to keep up to date, monitor: http://svn.debian.org/wsvn/nut/trunk/scripts/upower/95-upower-hid.rules'."\n\n"; print $outputUPower '# only support USB, else ignore'."\n".'SUBSYSTEM!="usb", GOTO="up_hid_end"'."\n\n"; print $outputUPower '# if usbraw device, ignore'."\n".'KERNEL!="hiddev*", GOTO="up_hid_end"'."\n\n"; print $outputUPower '# if an interface, ignore'."\n".'ENV{DEVTYPE}=="usb_interface", GOTO="up_hid_end"'."\n\n"; # Device scanner header open my $outputDevScanner, ">$outputDevScanner" || die "error $outputDevScanner : $!"; print $outputDevScanner '/* nutscan-usb'.$GPL_header."\n */\n\n"; print $outputDevScanner "#ifndef DEVSCAN_USB_H\n#define DEVSCAN_USB_H\n\n"; print $outputDevScanner "#include \n"; print $outputDevScanner "#include \"nut_stdint.h\"\t/* for uint16_t */\n\n"; # vid, pid, driver print $outputDevScanner "typedef struct {\n\tuint16_t\tvendorID;\n\tuint16_t\tproductID;\n\tchar*\tdriver_name;\n} usb_device_id_t;\n\n"; print $outputDevScanner "/* USB IDs device table */\nstatic usb_device_id_t usb_device_table[] = {\n\n"; # generate the file in alphabetical order (first for VendorID, then for ProductID) foreach my $vendorId (sort { lc $a cmp lc $b } keys %vendorName) { # HAL vendor header if ($vendorName{$vendorId}) { print $outHAL "\n \n"; } print $outHAL " \n"; # Hotplug vendor header if ($vendorName{$vendorId}) { print $outHotplug "\n# ".$vendorName{$vendorId}."\n"; } # udev vendor header if ($vendorName{$vendorId}) { print $outUdev "\n# ".$vendorName{$vendorId}."\n"; } # UPower vendor header flag $upowerMfrHeaderDone = 0; foreach my $productId (sort { lc $a cmp lc $b } keys %{$vendor{$vendorId}}) { # HAL device entry print $outHAL " \n"; print $outHAL " \n"; print $outHAL ' battery'."\n"; print $outHAL ' battery'."\n"; print $outHAL " hald-addon-".$vendor{$vendorId}{$productId}{"driver"}."\n"; print $outHAL ' ups'."\n"; print $outHAL ' '."\n"; # Hotplug device entry print $outHotplug "# ".$vendor{$vendorId}{$productId}{"comment"}."\n"; print $outHotplug "libhidups 0x0003 ".$vendorId." ".$productId." 0x0000 0x0000 0x00"; print $outHotplug " 0x00 0x00 0x00 0x00 0x00 0x00000000\n"; # udev device entry print $outUdev "# ".$vendor{$vendorId}{$productId}{"comment"}.' - '.$vendor{$vendorId}{$productId}{"driver"}."\n"; print $outUdev "ATTR{idVendor}==\"".removeHexPrefix($vendorId); print $outUdev "\", ATTR{idProduct}==\"".removeHexPrefix($productId)."\","; print $outUdev ' MODE="664", GROUP="@RUN_AS_GROUP@"'."\n"; # UPower device entry (only for USB/HID devices!) if ($vendor{$vendorId}{$productId}{"driver"} eq "usbhid-ups") { if ($upowerMfrHeaderDone == 0) { # UPower vendor header if ($vendorName{$vendorId}) { $tmpOutputUPower = $tmpOutputUPower."\n# ".$vendorName{$vendorId}."\n"; } print $outputUPower "ATTRS{idVendor}==\"".removeHexPrefix($vendorId)."\", ENV{UPOWER_VENDOR}=\"".$vendorName{$vendorId}."\"\n"; $upowerMfrHeaderDone = 1; } $tmpOutputUPower = $tmpOutputUPower."ATTRS{idVendor}==\"".removeHexPrefix($vendorId); $tmpOutputUPower = $tmpOutputUPower."\", ATTRS{idProduct}==\"".removeHexPrefix($productId)."\","; $tmpOutputUPower = $tmpOutputUPower.' ENV{UPOWER_BATTERY_TYPE}="ups"'."\n"; } # Device scanner entry print $outputDevScanner "\t{ ".$vendorId.', '.$productId.", \"".$vendor{$vendorId}{$productId}{"driver"}."\" },\n"; } # HAL vendor footer print $outHAL " \n"; } # HAL footer print $outHAL " \n"; print $outHAL " \n"; print $outHAL "\n"; # Udev footer print $outUdev "\n".'LABEL="nut-usbups_rules_end"'."\n"; # UPower... # ...flush device table print $outputUPower $tmpOutputUPower; # ...and print footer print $outputUPower "\n".'LABEL="up_hid_end"'."\n"; # Device scanner footer print $outputDevScanner "\t/* Terminating entry */\n\t{ -1, -1, NULL }\n};\n#endif /* DEVSCAN_USB_H */\n\n"; } sub find_usbdevs { return $File::Find::prune = 1 if $_ eq '.svn'; my $nameFile=$_; my $lastComment=""; open my $file,$nameFile or die "error open file $nameFile"; while(my $line=<$file>) { # catch comment (should permit comment on the precedent or on the current line of USB_DEVICE declaration) if($line =~/\s*\/\*(.+)\*\/\s*$/) { $lastComment=$1; } if($line =~/^\s*\{\s*USB_DEVICE\((.+)\,(.+)\)\s*/) # for example : { USB_DEVICE(MGE_VENDORID, 0x0001)... } { my $VendorID=trim($1); my $ProductID=trim($2); my $VendorName=""; # special thing for backward declaration using #DEFINE # Format: # /* vendor name */ # #define VENDORID 0x???? if(!($VendorID=~/\dx(\d|\w)+/)) { open my $fh,$nameFile or die "error open file $nameFile"; while(my $data=<$fh>) { # catch Vendor Name if($data =~/\s*\/\*(.+)\*\/\s*$/) { $VendorName=$1; } # catch VendorID if ($data =~ /(#define|#DEFINE)\s+$VendorID\s+(\dx(\d|\w)+)/) { $VendorID=$2; last; } } } # same thing for the productID if(!($ProductID=~/\dx(\d|\w)+/)) { my $data = do { open my $fh, $nameFile or die "error open file $nameFile"; join '', <$fh> }; if ($data =~ /(#define|#DEFINE)\s+$ProductID\s+(\dx(\d|\w)+)/) { $ProductID=$2; } else { die "In file $nameFile, for product $ProductID, can't find the declaration of the constant"; } } # store date (to be optimized) # and don't overwritte actual vendor names with empty values if( (!$vendorName{$VendorID}) or (($vendorName{$VendorID} eq "") and ($VendorName ne "")) ) { $vendorName{$VendorID}=trim($VendorName); } $vendor{$VendorID}{$ProductID}{"comment"}=$lastComment; # process the driver name my $driver=$nameFile; if($nameFile=~/(.+)-hid\.c/) { $driver="usbhid-ups"; } # FIXME: make a generic matching rule *.c => * elsif ($nameFile eq "bcmxcp_usb.c") { $driver="bcmxcp_usb"; } elsif ($nameFile eq "tripplite_usb.c") { $driver="tripplite_usb"; } elsif ($nameFile eq "blazer_usb.c") { $driver="blazer_usb"; } elsif ($nameFile eq "richcomm_usb.c") { $driver="richcomm_usb"; } else { die "Unknown driver type: $nameFile"; } $vendor{$VendorID}{$ProductID}{"driver"}=$driver; } } } sub removeHexPrefix { # make a local copy, not to alter the original entry my $string = $_[0]; $string =~ s/0x//; return $string; } sub trim { my($str) = shift =~ m!^\s*(.+?)\s*$!i; defined $str ? return $str : return ''; }