diff options
Diffstat (limited to 'perl-install/printer/detect.pm')
-rw-r--r-- | perl-install/printer/detect.pm | 637 |
1 files changed, 0 insertions, 637 deletions
diff --git a/perl-install/printer/detect.pm b/perl-install/printer/detect.pm deleted file mode 100644 index d02e89ac3..000000000 --- a/perl-install/printer/detect.pm +++ /dev/null @@ -1,637 +0,0 @@ -package printer::detect; - -use strict; -use common; -use modules; -use detect_devices; -use printer::data; - -sub local_detect() { - modules::any_conf->read->get_probeall("usb-interface") and eval { modules::load($usbprintermodule) }; - # Reload parallel port modules only when we were not called by - # automatic setup of print queues, to avoid recursive calls - if ($::autoqueue) { - whatUsbport(); - } else { - eval { modules::unload(qw(lp parport_pc ppdev parport)) }; #- on kernel 2.4 parport has to be unloaded to probe again - eval { modules::load(qw(ppdev parport_pc lp)) }; #- take care as not available on 2.4 kernel (silent error). - whatPrinter(); - } -} - -sub net_detect { whatNetPrinter(1, 0, @_) } - -sub net_smb_detect { whatNetPrinter(0, 1, @_) } - -sub detect { - local_detect(), whatNetPrinter(1, 1, @_); -} - - -#-CLASS:PRINTER; -#-MODEL:HP LaserJet 1100; -#-MANUFACTURER:Hewlett-Packard; -#-DESCRIPTION:HP LaserJet 1100 Printer; -#-COMMAND SET:MLC,PCL,PJL; -sub whatPrinter() { - my @res = (whatParport(), whatUsbport()); - grep { $_->{val}{CLASS} eq "PRINTER" } @res; -} - -sub whatParport() { - my @res; - my $i = 0; - foreach (sort { $a =~ /(\d+)/; my $m = $1; $b =~ /(\d+)/; my $n = $1; $m <=> $n } `ls -1d /proc/parport/[0-9]* /proc/sys/dev/parport/parport[0-9]* 2>/dev/null`) { - chomp; - my $elem = {}; - my $F; - open $F, "$_/autoprobe" or next; - { - local $_; - my $itemfound = 0; - while (<$F>) { - chomp; - if (/(.*):(.*);/) { #-# - $elem->{$1} = $2; - $elem->{$1} =~ s/Hewlett[-\s_]Packard/HP/; - $elem->{$1} =~ s/HEWLETT[-\s_]PACKARD/HP/; - $itemfound = 1; - # Add IEEE-1284 device ID string - $elem->{IEEE1284} .= $_; - } - } - # Some parallel printers miss the "CLASS" field - $elem->{CLASS} = 'PRINTER' - if $itemfound && !defined($elem->{CLASS}); - } - push @res, { port => "/dev/lp$i", val => $elem }; - $i ++; - } - @res; -} - -sub whatPrinterPort() { - grep { detect_devices::tryWrite($_) } qw(/dev/lp0 /dev/lp1 /dev/lp2 /dev/usb/lp0 /dev/usb/lp1 /dev/usb/lp2 /dev/usb/lp3 /dev/usb/lp4 /dev/usb/lp5 /dev/usb/lp6 /dev/usb/lp7 /dev/usb/lp8 /dev/usb/lp9); -} - -sub whatUsbport() { - # The printer manufacturer and model names obtained with the usb_probe() - # function were very messy, once there was a lot of noise around the - # manufacturers name ("Inc.", "SA", "International", ...) and second, - # all Epson inkjets answered with the name "Epson Stylus Color 760" which - # lead many newbies to install their Epson Stylus Photo XXX as an Epson - # Stylus Color 760 ... - # - # This routine based on an ioctl request gives very clean and correct - # manufacturer and model names, so that they are easily matched to the - # printer entries in the Foomatic database - my @res; - foreach my $i (0..15) { - my $port = "/dev/usb/lp$i"; - my $realport = devices::make($port); - next if !$realport; - next if ! -r $realport; - foreach my $j (1..3) { - open(my $PORT, $realport) or next; - my $idstr = ""; - # Calculation of IOCTL function 0x84005001 (to get device ID - # string): - # len = 1024 - # IOCNR_GET_DEVICE_ID = 1 - # LPIOC_GET_DEVICE_ID(len) = - # _IOC(_IOC_READ, 'P', IOCNR_GET_DEVICE_ID, len) - # _IOC(), _IOC_READ as defined in /usr/include/asm/ioctl.h - # Use "eval" so that program does not stop when IOCTL fails - eval { - my $output = "\0" x 1024; - ioctl($PORT, 0x84005001, $output); - $idstr = $output; - } or do { - close $PORT; - next; - }; - close $PORT; - # Cut resulting string to its real length - my $length = ord(substr($idstr, 1, 1)) + - (ord(substr($idstr, 0, 1)) << 8); - $idstr = substr($idstr, 2, $length-2); - # Remove non-printable characters - $idstr =~ tr/[\x00-\x1f]/./; - # If we do not find any item in the ID string, we try to read - # it again - my $itemfound = 0; - # Extract the printer data from the ID string - my ($manufacturer, $model, $serialnumber, $description, $commandset) = - ("", "", "", "", ""); - my ($sku); - if ($idstr =~ /CLS:([^;]+);/ || $idstr =~ /CLASS:([^;]+);/) { - $itemfound = 1; - } - if ($idstr =~ /MFG:([^;]+);/ || $idstr =~ /MANUFACTURER:([^;]+);/) { - $manufacturer = $1; - $manufacturer =~ s/Hewlett[-\s_]Packard/HP/; - $manufacturer =~ s/HEWLETT[-\s_]PACKARD/HP/; - $itemfound = 1; - } - # For HP's multi-function devices the real model name is in the "SKU" - # field. So use this field with priority for $model when it exists. - if ($idstr =~ /MDL:([^;]+);/ || $idstr =~ /MODEL:([^;]+);/) { - $model ||= $1; - $itemfound = 1; - } - if ($idstr =~ /SKU:([^;]+);/) { - $sku = $1; - $itemfound = 1; - } - if ($idstr =~ /DES:([^;]+);/ || $idstr =~ /DESCRIPTION:([^;]+);/) { - $description = $1; - $description =~ s/Hewlett[-\s_]Packard/HP/; - $description =~ s/HEWLETT[-\s_]PACKARD/HP/; - $itemfound = 1; - } - if (($idstr =~ /SE*R*N:([^;]+);/) || - ($idstr =~ /SN:([^;]+);/)) { - $serialnumber = $1; - $itemfound = 1; - } - if ($idstr =~ /CMD:([^;]+);/ || - $idstr =~ /COMMAND\s*SET:([^;]+);/) { - $commandset ||= $1; - $itemfound = 1; - } - # Nothing found? Try again if not in the third attempt, - # after the third attempt give up - next if !$itemfound; - # Was there a manufacturer and a model in the string? - if ($manufacturer eq "" || $model eq "") { - $manufacturer = ""; - $model = N("Unknown model"); - } - # No description field? Make one out of manufacturer and model. - if ($description eq "") { - $description = "$manufacturer $model"; - } - # Store this auto-detection result in the data structure - push @res, { port => $port, val => - { CLASS => 'PRINTER', - MODEL => $model, - MANUFACTURER => $manufacturer, - DESCRIPTION => $description, - SERIALNUMBER => $serialnumber, - 'COMMAND SET' => $commandset, - SKU => $sku, - IEEE1284 => $idstr, - } }; - last; - } - } - @res; -} - -sub whatNetPrinter { - my ($network, $smb, $timeout) = @_; - - my (@res); - - # Set timeouts for "nmap" - $timeout = 4000 if !$timeout; - my $irtimeout = $timeout / 2; - - # Which ports should be scanned? - my @portstoscan; - push @portstoscan, "139" if $smb; - push @portstoscan, "4010", "4020", "4030", "5503", "9100-9104" if $network; - - return () if $#portstoscan < 0; - my $portlist = join ",", @portstoscan; - - # Which hosts should be scanned? - # (Applying nmap to a whole network is very time-consuming, - # because nmap waits for a certain timeout period on non-existing - # hosts, so we get a lists of existing hosts by pinging the - # broadcast addresses for existing hosts and then scanning only - # them, which is much faster) - my @hostips = getIPsInLocalNetworks(); - return () if $#hostips < 0; - my $hostlist = join " ", @hostips; - - # Scan network for printers, the timeout settings are there to avoid - # delays caused by machines blocking their ports with a firewall - local *F; - open F, ($::testing ? "" : "chroot $::prefix/ ") . - qq(/bin/sh -c "export LC_ALL=C; nmap -r -P0 --host_timeout $timeout --initial_rtt_timeout $irtimeout -p $portlist $hostlist" 2> /dev/null |) - or return @res; - my ($host, $ip, $port, $modelinfo, $namechecked) = ("", "", "", "", 0); - while (my $line = <F>) { - chomp $line; - - # head line of the report of a host with the ports in question open - if (($line =~ m/^\s*Interesting\s+ports\s+on\s+(\S*)\s*\((\S+)\)\s*:\s*$/i) || - ($line =~ m/^\s*Interesting\s+ports\s+on\s+(\S+)\s*:\s*$/i)) { - ($host, $ip) = ($1, $2); - $ip = $host if !$ip; - $host = $ip if $host eq ""; - $port = ""; - $namechecked = 0; - - undef $modelinfo; - - } elsif ($line =~ m!^\s*(\d+)/\S+\s+open\s+!i) { - next if $ip eq ""; - $port = $1; - - # Check integrity of the host name (work around DNS problems - # by using IP if host name is broken) - if (!$namechecked && ($host ne $ip)) { - $namechecked = 1; # Do not check more than once - my $packedip = gethostbyname("$host"); - my ($a,$b,$c,$d) = unpack('C4',$packedip); - my $ipfromdns = sprintf("%d.%d.%d.%d", $a, $b, $c, $d); - $host = $ip if $ip ne $ipfromdns; - } - # Now we have all info for one printer - # Store this auto-detection result in the data structure - - # Determine the protocol by the port number - - # SMB/Windows - if ($port eq "139") { - my @shares = getSMBPrinterShares($ip); - foreach my $share (@shares) { - push @res, { port => "smb://$host/$share->{name}", - val => { CLASS => 'PRINTER', - MODEL => N("Unknown model"), - MANUFACTURER => "", - DESCRIPTION => $share->{description}, - SERIALNUMBER => "" - } - }; - } - } else { - if (!defined($modelinfo)) { - # SNMP request to auto-detect model - $modelinfo = getSNMPModel($ip); - } - if (defined($modelinfo)) { - push @res, { port => "socket://$host:$port", - val => $modelinfo - }; - } - } - } - } - close F; - @res; -} - -sub getNetworkInterfaces() { - - # subroutine determines the list of all network interfaces reported - # by "ifconfig", except "lo". - - # Return an empty list if no network is running - return () unless network_running(); - - my @interfaces; - - local *IFCONFIG_OUT; - open IFCONFIG_OUT, ($::testing ? "" : "chroot $::prefix/ ") . - '/bin/sh -c "export LC_ALL=C; ifconfig" 2> /dev/null |' or return (); - while (my $readline = <IFCONFIG_OUT>) { - # New entry ... - if ($readline =~ /^(\S+)\s/) { - my $dev = $1; - if ($dev ne "lo") { - push @interfaces, $dev; - } - } - } - close(IFCONFIG_OUT); - - @interfaces; -} - -sub getIPsOfLocalMachine() { - - # subroutine determines all IPs which point to the local machine, - # except 127.0.0.1 (localhost). - - # Return an empty list if no network is running - return () unless network_running(); - - # Read the output of "ifconfig" to determine the broadcast addresses of - # the local networks - my $dev_is_realnet = 0; - my @local_ips; - my $current_ip = ""; - - local *IFCONFIG_OUT; - open IFCONFIG_OUT, ($::testing ? "" : "chroot $::prefix/ ") . - '/bin/sh -c "export LC_ALL=C; ifconfig" 2> /dev/null |' or return (); - while (my $readline = <IFCONFIG_OUT>) { - # New entry ... - if ($readline =~ /^(\S+)\s/) { - my $dev = $1; - # ... for a real network (not lo = localhost) - $dev_is_realnet = $dev ne 'lo'; - # delete previous address - $current_ip = ""; - } - # Are we in the important line now? - if ($readline =~ /\sinet addr:([0-9]+\.[0-9]+\.[0-9]+\.[0-9]+)\s/) { - # Rip out the IP address - $current_ip = $1; - - # Are we in an entry for a real network? - if ($dev_is_realnet) { - # Store current IP address - push @local_ips, $current_ip; - } - } - } - close(IFCONFIG_OUT); - @local_ips; -} - -sub getIPsInLocalNetworks() { - - # subroutine determines the list of all hosts reachable in the local - # networks by means of pinging the broadcast addresses. - - # Return an empty list if no network is running - return () unless network_running(); - - # Read the output of "ifconfig" to determine the broadcast addresses of - # the local networks - my $dev_is_localnet = 0; - my $local_nets = {}; - my $dev; - local *IFCONFIG_OUT; - open IFCONFIG_OUT, ($::testing ? "" : "chroot $::prefix/ ") . - '/bin/sh -c "export LC_ALL=C; ifconfig" 2> /dev/null |' or return (); - while (my $readline = <IFCONFIG_OUT>) { - # New entry ... - if ($readline =~ /^(\S+)\s/) { - $dev = $1; - # ... for a local network (eth = ethernet, - # vmnet = VMWare, - # ethernet card connected to ISP excluded)? - $dev_is_localnet = $dev =~ /^eth/ || $dev =~ /^vmnet/; - } - if ($dev_is_localnet) { - # Are we in the important line now? - if ($readline =~ /\saddr:([0-9]+\.[0-9]+\.[0-9]+\.[0-9]+)\s/) { - # Rip out the broadcast IP address - $local_nets->{$dev}{ip} = $1; - } - if ($readline =~ /\sBcast:([0-9]+\.[0-9]+\.[0-9]+\.[0-9]+)\s/) { - # Rip out the broadcast IP address - $local_nets->{$dev}{bcast} = $1; - } - if ($readline =~ /\sMask:([0-9]+\.[0-9]+\.[0-9]+\.[0-9]+)\s/) { - # Rip out the broadcast IP address - $local_nets->{$dev}{mask} = $1; - } - } - } - close(IFCONFIG_OUT); - - # Now find all addresses in the local networks which we will investigate - my @addresses; - foreach my $dev (keys %{$local_nets}) { - my $ip = $local_nets->{$dev}{ip}; - my $bcast = $local_nets->{$dev}{bcast}; - my $mask = $local_nets->{$dev}{mask}; - if ($mask =~ /255.255.255.(\d+)/) { - # Small network, never more than 255 boxes, so we return - # all addresses belonging to this network, nwithout pinging - my $lastnumber = $1; - my $masknumber; - if ($lastnumber < 128) { - $masknumber = 24; - } elsif ($lastnumber < 192) { - $masknumber = 25; - } elsif ($lastnumber < 224) { - $masknumber = 26; - } elsif ($lastnumber < 240) { - $masknumber = 27; - } elsif ($lastnumber < 248) { - $masknumber = 28; - } elsif ($lastnumber < 252) { - $masknumber = 29; - } elsif ($lastnumber < 254) { - $masknumber = 30; - } elsif ($lastnumber < 255) { - $masknumber = 31; - } else { - $masknumber = 32; - } - push @addresses, "$ip/$masknumber"; - } else { - # Big network, probably more than 255 boxes, so ping the - # broadcast address and additionally "nmblookup" the - # networks (to find Windows servers which do not answer to ping) - local *F; - open F, ($::testing ? "" : "chroot $::prefix/ ") . - qq(/bin/sh -c "export LC_ALL=C; ping -w 1 -b -n $bcast 2> /dev/null | cut -f 4 -d ' ' | sed s/:// | egrep '^[0-9]+\\.[0-9]+\\.[0-9]+\\.[0-9]+' | uniq | sort" |) - or next; - local $_; - while (<F>) { chomp; push @addresses, $_ } - close F; - if (-x "/usr/bin/nmblookup") { - local *F; - open F, ($::testing ? "" : "chroot $::prefix/ ") . - qq(/bin/sh -c "export LC_ALL=C; nmblookup -B $bcast \\* 2> /dev/null | cut -f 1 -d ' ' | egrep '^[0-9]+\\.[0-9]+\\.[0-9]+\\.[0-9]+' | uniq | sort" |) - or next; - local $_; - while (<F>) { - chomp; - push @addresses, $_ if !(member($_,@addresses)); - } - } - } - } - - @addresses; -} - -sub getSMBPrinterShares { - my ($host) = @_; - - # SMB request to auto-detect shares - local *F; - open F, ($::testing ? "" : "chroot $::prefix/ ") . - qq(/bin/sh -c "export LC_ALL=C; smbclient -N -L $host" 2> /dev/null |) or return (); - my $insharelist = 0; - my @shares; - while (my $l = <F>) { - chomp $l; - if ($l =~ /^\s*Sharename\s+Type\s+Comment\s*$/i) { - $insharelist = 1; - } elsif ($l =~ /^\s*Server\s+Comment\s*$/i) { - $insharelist = 0; - } elsif ($l =~ /^\s*(\S+)\s+Printer\s*(.*)$/i && - $insharelist) { - my $name = $1; - my $description = $2; - $description =~ s/^(\s*)//; - push @shares, { name => $name, description => $description }; - } - } - close F; - - return @shares; -} - -sub getSNMPModel { - my ($host) = @_; - my $manufacturer = ""; - my $model = ""; - my $description = ""; - my $serialnumber = ""; - - # SNMP request to auto-detect model - local *F; - open F, ($::testing ? $::prefix : "chroot $::prefix/ ") . - qq(/bin/sh -c "scli -v 1 -c 'show printer info' $host" 2> /dev/null |) or - return { CLASS => 'PRINTER', - MODEL => N("Unknown model"), - MANUFACTURER => "", - DESCRIPTION => "", - SERIALNUMBER => "" - }; - while (my $l = <F>) { - chomp $l; - if ($l =~ /^\s*Manufacturer:\s*(\S.*)$/i && - $l =~ /^\s*Vendor:\s*(\S.*)$/i) { - $manufacturer = $1; - $manufacturer =~ s/Hewlett[-\s_]Packard/HP/; - $manufacturer =~ s/HEWLETT[-\s_]PACKARD/HP/; - } elsif ($l =~ /^\s*Model:\s*(\S.*)$/i) { - $model = $1; - } elsif ($l =~ /^\s*Description:\s*(\S.*)$/i) { - $description = $1; - $description =~ s/Hewlett[-\s_]Packard/HP/; - $description =~ s/HEWLETT[-\s_]PACKARD/HP/; - } elsif ($l =~ /^\s*Serial\s*Number:\s*(\S.*)$/i) { - $serialnumber = $1; - } - } - close F; - - # Was there a manufacturer and a model in the output? - # If not, get them from the description - if ($manufacturer eq "" || $model eq "") { - # Replace bad description - if ((length($description) < 5) && - (length($description) >= 5)) { - $description = $model; - } - # Guess manufacturer by model name - if ($description =~ - /^\s*(DeskJet|LaserJet|OfficeJet|PSC|PhotoSmart)\b/i) { - # HP printer - $manufacturer = "HP"; - $model = $description; - } elsif ($description =~ - /^\s*(Stylus|EPL|AcuLaser)\b/i) { - # Epson printer - $manufacturer = "Epson"; - $model = $description; - } elsif ($description =~ - /^\s*(Aficio)\b/i) { - # Ricoh printer - $manufacturer = "Ricoh"; - $model = $description; - } elsif ($description =~ - /^\s*(Optra|Color\s+JetPrinter)\b/i) { - # Lexmark printer - $manufacturer = "Lexmark"; - $model = $description; - } elsif ($description =~ - /^\s*(imageRunner|Pixma|Pixus|BJC|LBP)\b/i) { - # Canon printer - $manufacturer = "Canon"; - $model = $description; - } elsif ($description =~ - /^\s*(Phaser|DocuPrint|(Work|Document)\s*(Home|)Centre)\b/i) { - # Xerox printer - $manufacturer = "Xerox"; - $model = $description; - } elsif ($description =~ /^\s*(\S*)\s+(\S.*)$/) { - $manufacturer = $1 if $manufacturer eq ""; - $model = $2 if $model eq ""; - } - # No description field? Make one out of manufacturer and model. - } elsif ($description eq "") { - $description = "$manufacturer $model"; - } - - # We couldn't determine a model - $model = N("Unknown model") if $model eq ""; - - # Remove trailing spaces - $manufacturer =~ s/(\S+)\s+$/$1/; - $model =~ s/(\S+)\s+$/$1/; - $description =~ s/(\S+)\s+$/$1/; - $serialnumber =~ s/(\S+)\s+$/$1/; - - # Now we have all info for one printer - # Store this auto-detection result in the data structure - return { CLASS => 'PRINTER', - MODEL => $model, - MANUFACTURER => $manufacturer, - DESCRIPTION => $description, - SERIALNUMBER => $serialnumber - }; -} - -sub network_running() { - # If the network is not running return 0, otherwise 1. - local *F; - open F, ($::testing ? $::prefix : "chroot $::prefix/ ") . - '/bin/sh -c "export LC_ALL=C; /sbin/ifconfig" 2> /dev/null |' or - die 'Could not run "ifconfig"!'; - while (my $line = <F>) { - if ($line !~ /^lo\s+/ && # The loopback device can have been - # started by the spooler's startup script - $line =~ /^(\S+)\s+/) { # In this line starts an entry for a - # running network - close F; - return 1; - } - } - close F; - return 0; -} - -sub parport_addr { - # auto-detect the parallel port addresses - my ($device) = @_; - $device =~ m!^/dev/lp(\d+)$! or - $device =~ m!^/dev/printers/(\d+)$!; - my $portnumber = $1; - my $i = 0; - my $parportdir; - foreach (sort { $a =~ /(\d+)/; my $m = $1; $b =~ /(\d+)/; my $n = $1; $m <=> $n } `ls -1d /proc/parport/[0-9]* /proc/sys/dev/parport/parport[0-9]* 2>/dev/null`) { - chomp; - if ($i == $portnumber) { - $parportdir = $_; - last; - } - $i++; - } - my $parport_addresses = - `cat $parportdir/base-addr`; - my $address_arg; - if ($parport_addresses =~ /^\s*(\d+)\s+(\d+)\s*$/) { - $address_arg = sprintf(" -base 0x%x -basehigh 0x%x", $1, $2); - } elsif ($parport_addresses =~ /^\s*(\d+)\s*$/) { - $address_arg = sprintf(" -base 0x%x", $1); - } else { - $address_arg = ""; - } - return $address_arg; -} - -1; |