usr
/
bin
/
Go to Home Directory
+
Upload
Create File
root@0UT1S:~$
Execute
By Order of Mr.0UT1S
[DIR] ..
N/A
7za
46 bytes
Rename
Delete
GET
15.84 KB
Rename
Delete
Mail
408.89 KB
Rename
Delete
[
53.67 KB
Rename
Delete
aclocal
35.62 KB
Rename
Delete
aclocal-1.16
35.62 KB
Rename
Delete
addr2line
33.41 KB
Rename
Delete
animate
11.84 KB
Rename
Delete
ar
61.96 KB
Rename
Delete
arch
37.41 KB
Rename
Delete
arpaname
11.82 KB
Rename
Delete
as
889.91 KB
Rename
Delete
aspell
159.50 KB
Rename
Delete
at
1.01 KB
Rename
Delete
atq
1.01 KB
Rename
Delete
atrm
1.02 KB
Rename
Delete
autoconf
14.42 KB
Rename
Delete
autoheader
8.33 KB
Rename
Delete
autom4te
31.43 KB
Rename
Delete
automake
251.90 KB
Rename
Delete
automake-1.16
251.90 KB
Rename
Delete
autoreconf
20.57 KB
Rename
Delete
autoscan
16.72 KB
Rename
Delete
autoupdate
33.08 KB
Rename
Delete
awk
669.77 KB
Rename
Delete
b2sum
57.76 KB
Rename
Delete
base32
41.55 KB
Rename
Delete
base64
41.56 KB
Rename
Delete
basename
37.49 KB
Rename
Delete
bash
1.10 MB
Rename
Delete
bashbug-64
7.18 KB
Rename
Delete
batch
137 bytes
Rename
Delete
bison
437.72 KB
Rename
Delete
bunzip2
36.86 KB
Rename
Delete
bzcat
36.86 KB
Rename
Delete
bzcmp
2.08 KB
Rename
Delete
bzdiff
2.08 KB
Rename
Delete
bzgrep
1.64 KB
Rename
Delete
bzip2
36.86 KB
Rename
Delete
bzip2recover
16.44 KB
Rename
Delete
bzless
1.23 KB
Rename
Delete
bzmore
1.23 KB
Rename
Delete
c++
1.21 MB
Rename
Delete
c++filt
28.89 KB
Rename
Delete
c89
224 bytes
Rename
Delete
c99
215 bytes
Rename
Delete
cagefs_enter.proxied
1.03 KB
Rename
Delete
cal
65.98 KB
Rename
Delete
captoinfo
85.31 KB
Rename
Delete
cat
37.54 KB
Rename
Delete
catchsegv
3.21 KB
Rename
Delete
cc
1.21 MB
Rename
Delete
chcon
70.43 KB
Rename
Delete
chgrp
66.35 KB
Rename
Delete
chmod
62.29 KB
Rename
Delete
chown
70.39 KB
Rename
Delete
chrt
37.18 KB
Rename
Delete
cksum
37.46 KB
Rename
Delete
cldetect
10.36 KB
Rename
Delete
clear
12.54 KB
Rename
Delete
clusterdb
70.23 KB
Rename
Delete
cmp
103.76 KB
Rename
Delete
col
29.00 KB
Rename
Delete
colcrt
16.48 KB
Rename
Delete
colrm
24.88 KB
Rename
Delete
column
49.47 KB
Rename
Delete
comm
41.63 KB
Rename
Delete
compare
11.85 KB
Rename
Delete
composite
11.84 KB
Rename
Delete
conjure
11.84 KB
Rename
Delete
convert
11.84 KB
Rename
Delete
cp
148.05 KB
Rename
Delete
cpan
8.17 KB
Rename
Delete
cpp
1.21 MB
Rename
Delete
createdb
70.22 KB
Rename
Delete
createuser
74.63 KB
Rename
Delete
crontab
1.36 KB
Rename
Delete
crontab.cagefs
54.16 KB
Rename
Delete
csplit
53.76 KB
Rename
Delete
curl
230.09 KB
Rename
Delete
cut
49.59 KB
Rename
Delete
date
106.03 KB
Rename
Delete
dbiprof
6.06 KB
Rename
Delete
dd
78.05 KB
Rename
Delete
delv
42.46 KB
Rename
Delete
df
91.16 KB
Rename
Delete
diff
268.01 KB
Rename
Delete
diff3
128.60 KB
Rename
Delete
dig
162.19 KB
Rename
Delete
dir
139.97 KB
Rename
Delete
dircolors
49.63 KB
Rename
Delete
dirname
33.44 KB
Rename
Delete
display
11.84 KB
Rename
Delete
dnstap-read
20.43 KB
Rename
Delete
dropdb
66.02 KB
Rename
Delete
dropuser
65.99 KB
Rename
Delete
du
107.10 KB
Rename
Delete
easy_install-3
bytes
Rename
Delete
echo
37.43 KB
Rename
Delete
egrep
28 bytes
Rename
Delete
enc2xs
40.97 KB
Rename
Delete
enchant
21.08 KB
Rename
Delete
enchant-lsmod
13.09 KB
Rename
Delete
env
41.43 KB
Rename
Delete
eps2eps
639 bytes
Rename
Delete
eqn
232.16 KB
Rename
Delete
ex
1.13 MB
Rename
Delete
expand
41.66 KB
Rename
Delete
expr
49.65 KB
Rename
Delete
factor
86.05 KB
Rename
Delete
false
33.39 KB
Rename
Delete
fc-cache
132 bytes
Rename
Delete
fc-cache-64
20.35 KB
Rename
Delete
fc-cat
16.35 KB
Rename
Delete
fc-conflist
12.25 KB
Rename
Delete
fc-list
12.25 KB
Rename
Delete
fc-match
16.26 KB
Rename
Delete
fc-pattern
12.26 KB
Rename
Delete
fc-query
12.24 KB
Rename
Delete
fc-scan
12.26 KB
Rename
Delete
fc-validate
16.26 KB
Rename
Delete
fgrep
28 bytes
Rename
Delete
file
24.68 KB
Rename
Delete
find
223.30 KB
Rename
Delete
flex
428.45 KB
Rename
Delete
flex++
428.45 KB
Rename
Delete
flock
33.20 KB
Rename
Delete
fmt
45.57 KB
Rename
Delete
fold
41.48 KB
Rename
Delete
free
20.79 KB
Rename
Delete
freetype-config
4.31 KB
Rename
Delete
funzip
36.63 KB
Rename
Delete
g++
1.21 MB
Rename
Delete
gawk
669.77 KB
Rename
Delete
gcc
1.21 MB
Rename
Delete
gcc-ar
36.66 KB
Rename
Delete
gcc-nm
36.66 KB
Rename
Delete
gcc-ranlib
36.66 KB
Rename
Delete
gcov
1.31 MB
Rename
Delete
gcov-dump
570.88 KB
Rename
Delete
gcov-tool
607.75 KB
Rename
Delete
gem
542 bytes
Rename
Delete
gencat
24.84 KB
Rename
Delete
geoiplookup
21.89 KB
Rename
Delete
geoiplookup6
21.65 KB
Rename
Delete
geqn
232.16 KB
Rename
Delete
getconf
32.46 KB
Rename
Delete
getent
33.13 KB
Rename
Delete
getopt
20.52 KB
Rename
Delete
ghostscript
12.35 KB
Rename
Delete
git
3.67 MB
Rename
Delete
git-receive-pack
3.67 MB
Rename
Delete
git-shell
2.13 MB
Rename
Delete
git-upload-archive
3.67 MB
Rename
Delete
git-upload-pack
3.67 MB
Rename
Delete
gm
7.82 KB
Rename
Delete
gmake
235.32 KB
Rename
Delete
gneqn
908 bytes
Rename
Delete
gnroff
3.23 KB
Rename
Delete
gpg
1.04 MB
Rename
Delete
gpg-agent
419.29 KB
Rename
Delete
gpg-error
34.16 KB
Rename
Delete
gpg-zip
3.44 KB
Rename
Delete
gpgsplit
87.02 KB
Rename
Delete
gpgv
451.58 KB
Rename
Delete
gpic
293.84 KB
Rename
Delete
gprof
103.36 KB
Rename
Delete
grep
193.63 KB
Rename
Delete
groff
124.92 KB
Rename
Delete
grops
191.14 KB
Rename
Delete
grotty
141.90 KB
Rename
Delete
groups
37.47 KB
Rename
Delete
gs
12.35 KB
Rename
Delete
gsnd
277 bytes
Rename
Delete
gsoelim
42.55 KB
Rename
Delete
gtar
449.03 KB
Rename
Delete
gtbl
154.61 KB
Rename
Delete
gtroff
805.02 KB
Rename
Delete
gunzip
2.29 KB
Rename
Delete
gzexe
6.23 KB
Rename
Delete
gzip
94.67 KB
Rename
Delete
h2ph
28.69 KB
Rename
Delete
h2xs
59.44 KB
Rename
Delete
head
45.58 KB
Rename
Delete
hexdump
57.50 KB
Rename
Delete
host
142.30 KB
Rename
Delete
hostid
33.41 KB
Rename
Delete
hostname
21.16 KB
Rename
Delete
hunspell
144.70 KB
Rename
Delete
iconv
61.44 KB
Rename
Delete
id
45.52 KB
Rename
Delete
identify
11.84 KB
Rename
Delete
idn
39.41 KB
Rename
Delete
ifnames
4.03 KB
Rename
Delete
import
11.84 KB
Rename
Delete
infocmp
61.05 KB
Rename
Delete
infotocap
85.31 KB
Rename
Delete
install
156.25 KB
Rename
Delete
instmodsh
4.10 KB
Rename
Delete
ionice
28.98 KB
Rename
Delete
ipcrm
28.99 KB
Rename
Delete
ipcs
53.39 KB
Rename
Delete
isosize
24.88 KB
Rename
Delete
ispell
988 bytes
Rename
Delete
join
53.77 KB
Rename
Delete
kill
37.27 KB
Rename
Delete
ld
1.71 MB
Rename
Delete
ld.bfd
1.71 MB
Rename
Delete
ldd
5.31 KB
Rename
Delete
less
173.76 KB
Rename
Delete
lessecho
12.40 KB
Rename
Delete
lesskey
21.99 KB
Rename
Delete
lesspipe.sh
3.07 KB
Rename
Delete
lex
428.45 KB
Rename
Delete
libnetcfg
15.41 KB
Rename
Delete
libtool
359.11 KB
Rename
Delete
libtoolize
126.17 KB
Rename
Delete
link
33.41 KB
Rename
Delete
ln
70.57 KB
Rename
Delete
locale
56.45 KB
Rename
Delete
localedef
307.47 KB
Rename
Delete
logger
49.98 KB
Rename
Delete
login
40.96 KB
Rename
Delete
logname
33.42 KB
Rename
Delete
look
16.45 KB
Rename
Delete
ls
139.97 KB
Rename
Delete
lynx
1.84 MB
Rename
Delete
m4
185.56 KB
Rename
Delete
mail
408.89 KB
Rename
Delete
mailx
408.89 KB
Rename
Delete
make
235.32 KB
Rename
Delete
make-dummy-cert
610 bytes
Rename
Delete
mariadb
4.34 MB
Rename
Delete
mariadb-access
109.34 KB
Rename
Delete
mariadb-admin
3.88 MB
Rename
Delete
mariadb-binlog
4.14 MB
Rename
Delete
mariadb-check
3.88 MB
Rename
Delete
mariadb-dump
3.96 MB
Rename
Delete
mariadb-find-rows
3.21 KB
Rename
Delete
mariadb-import
3.87 MB
Rename
Delete
mariadb-show
3.87 MB
Rename
Delete
mariadb-waitpid
3.56 MB
Rename
Delete
mc
1.30 MB
Rename
Delete
mcdiff
1.30 MB
Rename
Delete
mcedit
1.30 MB
Rename
Delete
mcookie
33.26 KB
Rename
Delete
mcview
1.30 MB
Rename
Delete
md5sum
45.62 KB
Rename
Delete
mesg
16.36 KB
Rename
Delete
mkdir
82.79 KB
Rename
Delete
mkfifo
66.56 KB
Rename
Delete
mknod
70.55 KB
Rename
Delete
mktemp
45.73 KB
Rename
Delete
mogrify
11.84 KB
Rename
Delete
montage
11.84 KB
Rename
Delete
more
44.94 KB
Rename
Delete
msql2mysql
1.41 KB
Rename
Delete
mv
144.03 KB
Rename
Delete
my_print_defaults
3.56 MB
Rename
Delete
mysql
4.34 MB
Rename
Delete
mysql_config
4.60 KB
Rename
Delete
mysql_find_rows
3.21 KB
Rename
Delete
mysql_waitpid
3.56 MB
Rename
Delete
mysqlaccess
109.34 KB
Rename
Delete
mysqladmin
3.88 MB
Rename
Delete
mysqlbinlog
4.14 MB
Rename
Delete
mysqlcheck
3.88 MB
Rename
Delete
mysqldump
3.96 MB
Rename
Delete
mysqlimport
3.87 MB
Rename
Delete
mysqlshow
3.87 MB
Rename
Delete
namei
33.10 KB
Rename
Delete
nano
247.94 KB
Rename
Delete
neqn
908 bytes
Rename
Delete
nice
37.41 KB
Rename
Delete
nl
45.63 KB
Rename
Delete
nm
50.38 KB
Rename
Delete
nohup
37.48 KB
Rename
Delete
nproc
37.48 KB
Rename
Delete
nroff
3.23 KB
Rename
Delete
nslookup
146.26 KB
Rename
Delete
nsupdate
73.05 KB
Rename
Delete
numfmt
65.71 KB
Rename
Delete
objcopy
240.07 KB
Rename
Delete
objdump
419.76 KB
Rename
Delete
od
73.88 KB
Rename
Delete
openssl
745.95 KB
Rename
Delete
pango-list
11.88 KB
Rename
Delete
pango-view
57.44 KB
Rename
Delete
passwd
1.02 KB
Rename
Delete
paste
37.46 KB
Rename
Delete
patch
206.46 KB
Rename
Delete
pathchk
37.41 KB
Rename
Delete
pdf2dsc
698 bytes
Rename
Delete
pdf2ps
909 bytes
Rename
Delete
perl
12.44 KB
Rename
Delete
perl5.26.3
12.44 KB
Rename
Delete
perlbug
44.39 KB
Rename
Delete
perldoc
118 bytes
Rename
Delete
perlivp
10.56 KB
Rename
Delete
perlml
6.86 KB
Rename
Delete
perlthanks
44.39 KB
Rename
Delete
pg_dump
399.43 KB
Rename
Delete
pg_dumpall
107.11 KB
Rename
Delete
pg_restore
173.34 KB
Rename
Delete
pgrep
28.84 KB
Rename
Delete
php
937 bytes
Rename
Delete
pic
293.84 KB
Rename
Delete
piconv
8.08 KB
Rename
Delete
pinentry
2.35 KB
Rename
Delete
pinentry-curses
77.89 KB
Rename
Delete
ping
66.13 KB
Rename
Delete
pinky
41.53 KB
Rename
Delete
pip-3
bytes
Rename
Delete
pip3
bytes
Rename
Delete
pkg-config
40.04 KB
Rename
Delete
pkgconf
40.04 KB
Rename
Delete
pkill
28.84 KB
Rename
Delete
pl2pm
4.43 KB
Rename
Delete
pmap
32.78 KB
Rename
Delete
pod2html
4.04 KB
Rename
Delete
pod2man
14.68 KB
Rename
Delete
pod2text
10.55 KB
Rename
Delete
pod2usage
3.86 KB
Rename
Delete
podchecker
3.57 KB
Rename
Delete
podselect
2.47 KB
Rename
Delete
post-grohtml
238.73 KB
Rename
Delete
pr
82.23 KB
Rename
Delete
pre-grohtml
130.55 KB
Rename
Delete
precat
5.52 KB
Rename
Delete
preunzip
5.52 KB
Rename
Delete
prezip
5.52 KB
Rename
Delete
prezip-bin
11.98 KB
Rename
Delete
printenv
33.40 KB
Rename
Delete
printf
53.64 KB
Rename
Delete
prove
13.24 KB
Rename
Delete
ps
134.75 KB
Rename
Delete
ps2ascii
631 bytes
Rename
Delete
ps2epsi
2.69 KB
Rename
Delete
ps2pdf
272 bytes
Rename
Delete
ps2pdf12
215 bytes
Rename
Delete
ps2pdf13
215 bytes
Rename
Delete
ps2pdf14
215 bytes
Rename
Delete
ps2pdfwr
1.07 KB
Rename
Delete
ps2ps
647 bytes
Rename
Delete
ps2ps2
669 bytes
Rename
Delete
psql
644.33 KB
Rename
Delete
ptx
78.07 KB
Rename
Delete
pwd
37.50 KB
Rename
Delete
pwdx
12.68 KB
Rename
Delete
pydoc-3
bytes
Rename
Delete
pydoc3
bytes
Rename
Delete
python2
7.84 KB
Rename
Delete
python2.7
7.84 KB
Rename
Delete
python3
11.59 KB
Rename
Delete
python3.6
11.59 KB
Rename
Delete
python3.6m
11.59 KB
Rename
Delete
pyvenv-3
bytes
Rename
Delete
ranlib
61.98 KB
Rename
Delete
raw
16.49 KB
Rename
Delete
readelf
624.54 KB
Rename
Delete
readlink
45.96 KB
Rename
Delete
realpath
50.02 KB
Rename
Delete
recode
47.03 KB
Rename
Delete
reindexdb
70.32 KB
Rename
Delete
rename
16.50 KB
Rename
Delete
renew-dummy-cert
725 bytes
Rename
Delete
renice
16.46 KB
Rename
Delete
reset
24.76 KB
Rename
Delete
rev
12.45 KB
Rename
Delete
rm
70.47 KB
Rename
Delete
rmdir
45.54 KB
Rename
Delete
rnano
247.94 KB
Rename
Delete
rsync
510.11 KB
Rename
Delete
ruby
11.84 KB
Rename
Delete
run-with-aspell
85 bytes
Rename
Delete
runcon
37.45 KB
Rename
Delete
rvi
1.13 MB
Rename
Delete
rview
1.13 MB
Rename
Delete
rvim
2.93 MB
Rename
Delete
scalar
2.18 MB
Rename
Delete
scl
36.87 KB
Rename
Delete
scl_enabled
258 bytes
Rename
Delete
scl_source
1.82 KB
Rename
Delete
scp
102.85 KB
Rename
Delete
screen
482.46 KB
Rename
Delete
script
36.79 KB
Rename
Delete
sdiff
105.33 KB
Rename
Delete
sed
115.48 KB
Rename
Delete
selectorctl
7.63 KB
Rename
Delete
seq
53.52 KB
Rename
Delete
setsid
16.38 KB
Rename
Delete
setterm
45.12 KB
Rename
Delete
sftp
159.74 KB
Rename
Delete
sh
1.10 MB
Rename
Delete
sha1sum
45.63 KB
Rename
Delete
sha224sum
45.66 KB
Rename
Delete
sha256sum
45.66 KB
Rename
Delete
sha384sum
45.66 KB
Rename
Delete
sha512sum
45.66 KB
Rename
Delete
shred
61.94 KB
Rename
Delete
shuf
58.16 KB
Rename
Delete
size
33.25 KB
Rename
Delete
skill
28.80 KB
Rename
Delete
slabtop
20.84 KB
Rename
Delete
sleep
37.47 KB
Rename
Delete
snice
28.80 KB
Rename
Delete
soelim
42.55 KB
Rename
Delete
sort
123.55 KB
Rename
Delete
spell
122 bytes
Rename
Delete
splain
18.70 KB
Rename
Delete
split
58.13 KB
Rename
Delete
sprof
28.67 KB
Rename
Delete
sqlite3
1.28 MB
Rename
Delete
ssh
757.54 KB
Rename
Delete
ssh-add
346.13 KB
Rename
Delete
ssh-agent
325.58 KB
Rename
Delete
ssh-copy-id
10.44 KB
Rename
Delete
ssh-keygen
427.16 KB
Rename
Delete
ssh-keyscan
428.57 KB
Rename
Delete
stat
86.23 KB
Rename
Delete
stdbuf
49.58 KB
Rename
Delete
strace
1.94 MB
Rename
Delete
stream
11.83 KB
Rename
Delete
strings
37.43 KB
Rename
Delete
strip
240.09 KB
Rename
Delete
stty
77.68 KB
Rename
Delete
sum
45.61 KB
Rename
Delete
sync
37.43 KB
Rename
Delete
tabs
16.55 KB
Rename
Delete
tac
41.57 KB
Rename
Delete
tail
74.20 KB
Rename
Delete
tar
449.03 KB
Rename
Delete
taskset
37.25 KB
Rename
Delete
tbl
154.61 KB
Rename
Delete
tclsh
9.04 KB
Rename
Delete
tclsh8.6
9.04 KB
Rename
Delete
tee
41.55 KB
Rename
Delete
test
53.63 KB
Rename
Delete
tic
85.31 KB
Rename
Delete
timeout
41.93 KB
Rename
Delete
tload
16.76 KB
Rename
Delete
tmpwatch
35.47 KB
Rename
Delete
toe
16.45 KB
Rename
Delete
top
121.70 KB
Rename
Delete
touch
94.02 KB
Rename
Delete
tput
24.80 KB
Rename
Delete
tr
49.70 KB
Rename
Delete
traceroute
70.97 KB
Rename
Delete
troff
805.02 KB
Rename
Delete
true
33.40 KB
Rename
Delete
truncate
41.44 KB
Rename
Delete
tset
24.76 KB
Rename
Delete
tsort
41.57 KB
Rename
Delete
tty
33.39 KB
Rename
Delete
tzselect
15.01 KB
Rename
Delete
uapi
1.02 KB
Rename
Delete
ul
20.58 KB
Rename
Delete
uname
37.41 KB
Rename
Delete
unexpand
45.68 KB
Rename
Delete
uniq
49.72 KB
Rename
Delete
unlink
33.41 KB
Rename
Delete
unversioned-python
bytes
Rename
Delete
unzip
201.87 KB
Rename
Delete
unzipsfx
101.48 KB
Rename
Delete
uptime
12.59 KB
Rename
Delete
users
37.47 KB
Rename
Delete
utmpdump
28.66 KB
Rename
Delete
vacuumdb
78.46 KB
Rename
Delete
vdir
139.97 KB
Rename
Delete
vi
1.13 MB
Rename
Delete
view
1.13 MB
Rename
Delete
vim
2.93 MB
Rename
Delete
vimdiff
2.93 MB
Rename
Delete
vimtutor
2.07 KB
Rename
Delete
vmstat
36.79 KB
Rename
Delete
watch
29.19 KB
Rename
Delete
wc
49.72 KB
Rename
Delete
wget
521.41 KB
Rename
Delete
whereis
29.27 KB
Rename
Delete
which
29.44 KB
Rename
Delete
who
53.68 KB
Rename
Delete
whoami
33.41 KB
Rename
Delete
word-list-compress
11.99 KB
Rename
Delete
x86_64-redhat-linux-c++
1.21 MB
Rename
Delete
x86_64-redhat-linux-g++
1.21 MB
Rename
Delete
x86_64-redhat-linux-gcc
1.21 MB
Rename
Delete
x86_64-redhat-linux-gcc-8
1.21 MB
Rename
Delete
xargs
74.11 KB
Rename
Delete
xmlcatalog
20.38 KB
Rename
Delete
xmllint
73.37 KB
Rename
Delete
xmlwf
32.96 KB
Rename
Delete
xsltproc
28.47 KB
Rename
Delete
xsubpp
4.96 KB
Rename
Delete
xxd
20.52 KB
Rename
Delete
yes
33.45 KB
Rename
Delete
zcat
1.94 KB
Rename
Delete
zcmp
1.64 KB
Rename
Delete
zdiff
5.74 KB
Rename
Delete
zegrep
29 bytes
Rename
Delete
zfgrep
29 bytes
Rename
Delete
zforce
2.03 KB
Rename
Delete
zgrep
7.40 KB
Rename
Delete
zip
229.00 KB
Rename
Delete
zipcloak
102.91 KB
Rename
Delete
zipgrep
2.88 KB
Rename
Delete
zipinfo
201.87 KB
Rename
Delete
zipnote
97.76 KB
Rename
Delete
zipsplit
97.76 KB
Rename
Delete
zless
2.15 KB
Rename
Delete
zmore
1.80 KB
Rename
Delete
znew
4.45 KB
Rename
Delete
zsoelim
42.55 KB
Rename
Delete
#!/usr/bin/perl eval 'exec /usr/bin/perl -S $0 ${1+"$@"}' if $running_under_some_shell; BEGIN { pop @INC if $INC[-1] eq '.' } use strict; use Config; use File::Path qw(mkpath); use Getopt::Std; # Make sure read permissions for all are set: if (defined umask && (umask() & 0444)) { umask (umask() & ~0444); } getopts('Dd:rlhaQe'); use vars qw($opt_D $opt_d $opt_r $opt_l $opt_h $opt_a $opt_Q $opt_e); die "-r and -a options are mutually exclusive\n" if ($opt_r and $opt_a); my @inc_dirs = inc_dirs() if $opt_a; my $Exit = 0; my $Dest_dir = $opt_d || $Config{installsitearch}; die "Destination directory $Dest_dir doesn't exist or isn't a directory\n" unless -d $Dest_dir; my @isatype = qw( char uchar u_char short ushort u_short int uint u_int long ulong u_long FILE key_t caddr_t float double size_t ); my %isatype; @isatype{@isatype} = (1) x @isatype; my $inif = 0; my %Is_converted; my %bad_file = (); @ARGV = ('-') unless @ARGV; build_preamble_if_necessary(); sub reindent($) { my($text) = shift; $text =~ s/\n/\n /g; $text =~ s/ /\t/g; $text; } my ($t, $tab, %curargs, $new, $eval_index, $dir, $name, $args, $outfile); my ($incl, $incl_type, $incl_quote, $next); while (defined (my $file = next_file())) { if (-l $file and -d $file) { link_if_possible($file) if ($opt_l); next; } # Recover from header files with unbalanced cpp directives $t = ''; $tab = 0; # $eval_index goes into '#line' directives, to help locate syntax errors: $eval_index = 1; if ($file eq '-') { open(IN, "-"); open(OUT, ">-"); } else { ($outfile = $file) =~ s/\.h$/.ph/ || next; print "$file -> $outfile\n" unless $opt_Q; if ($file =~ m|^(.*)/|) { $dir = $1; mkpath "$Dest_dir/$dir"; } if ($opt_a) { # automagic mode: locate header file in @inc_dirs foreach (@inc_dirs) { chdir $_; last if -f $file; } } open(IN, "<", "$file") || (($Exit = 1),(warn "Can't open $file: $!\n"),next); open(OUT, ">", "$Dest_dir/$outfile") || die "Can't create $outfile: $!\n"; } print OUT "require '_h2ph_pre.ph';\n\n", "no warnings qw(redefine misc);\n\n"; while (defined (local $_ = next_line($file))) { if (s/^\s*\#\s*//) { if (s/^define\s+(\w+)//) { $name = $1; $new = ''; s/\s+$//; s/\(\w+\s*\(\*\)\s*\(\w*\)\)\s*(-?\d+)/$1/; # (int (*)(foo_t))0 if (s/^\(([\w,\s]*)\)//) { $args = $1; my $proto = '() '; if ($args ne '') { $proto = ''; foreach my $arg (split(/,\s*/,$args)) { $arg =~ s/^\s*([^\s].*[^\s])\s*$/$1/; $curargs{$arg} = 1; } $args =~ s/\b(\w)/\$$1/g; $args = "my($args) = \@_;\n$t "; } s/^\s+//; expr(); $new =~ s/(["\\])/\\$1/g; #"]); EMIT($proto); } else { s/^\s+//; expr(); $new = 1 if $new eq ''; # Shunt around such directives as '#define FOO FOO': next if $new =~ /^\s*&\Q$name\E\s*\z/; $new = reindent($new); $args = reindent($args); $new =~ s/(['\\])/\\$1/g; #']); print OUT $t, 'eval '; if ($opt_h) { print OUT "\"\\n#line $eval_index $outfile\\n\" . "; $eval_index++; } print OUT "'sub $name () {$new;}' unless defined(&$name);\n"; } } elsif (/^(include|import|include_next)\s*([<\"])(.*)[>\"]/) { $incl_type = $1; $incl_quote = $2; $incl = $3; if (($incl_type eq 'include_next') || ($opt_e && exists($bad_file{$incl}))) { $incl =~ s/\.h$/.ph/; print OUT ($t, "eval {\n"); $tab += 4; $t = "\t" x ($tab / 8) . ' ' x ($tab % 8); print OUT ($t, "my(\@REM);\n"); if ($incl_type eq 'include_next') { print OUT ($t, "my(\%INCD) = map { \$INC{\$_} => 1 } ", "(grep { \$_ eq \"$incl\" } ", "keys(\%INC));\n"); print OUT ($t, "\@REM = map { \"\$_/$incl\" } ", "(grep { not exists(\$INCD{\"\$_/$incl\"})", " and -f \"\$_/$incl\" } \@INC);\n"); } else { print OUT ($t, "\@REM = map { \"\$_/$incl\" } ", "(grep {-r \"\$_/$incl\" } \@INC);\n"); } print OUT ($t, "require \"\$REM[0]\" if \@REM;\n"); $tab -= 4; $t = "\t" x ($tab / 8) . ' ' x ($tab % 8); print OUT ($t, "};\n"); print OUT ($t, "warn(\$\@) if \$\@;\n"); } else { $incl =~ s/\.h$/.ph/; # copy the prefix in the quote syntax (#include "x.h") case if ($incl !~ m|/| && $incl_quote eq q{"} && $file =~ m|^(.*)/|) { $incl = "$1/$incl"; } print OUT $t,"require '$incl';\n"; } } elsif (/^ifdef\s+(\w+)/) { print OUT $t,"if(defined(&$1)) {\n"; $tab += 4; $t = "\t" x ($tab / 8) . ' ' x ($tab % 8); } elsif (/^ifndef\s+(\w+)/) { print OUT $t,"unless(defined(&$1)) {\n"; $tab += 4; $t = "\t" x ($tab / 8) . ' ' x ($tab % 8); } elsif (s/^if\s+//) { $new = ''; $inif = 1; expr(); $inif = 0; print OUT $t,"if($new) {\n"; $tab += 4; $t = "\t" x ($tab / 8) . ' ' x ($tab % 8); } elsif (s/^elif\s+//) { $new = ''; $inif = 1; expr(); $inif = 0; $tab -= 4; $t = "\t" x ($tab / 8) . ' ' x ($tab % 8); print OUT $t,"}\n elsif($new) {\n"; $tab += 4; $t = "\t" x ($tab / 8) . ' ' x ($tab % 8); } elsif (/^else/) { $tab -= 4; $t = "\t" x ($tab / 8) . ' ' x ($tab % 8); print OUT $t,"} else {\n"; $tab += 4; $t = "\t" x ($tab / 8) . ' ' x ($tab % 8); } elsif (/^endif/) { $tab -= 4; $t = "\t" x ($tab / 8) . ' ' x ($tab % 8); print OUT $t,"}\n"; } elsif(/^undef\s+(\w+)/) { print OUT $t, "undef(&$1) if defined(&$1);\n"; } elsif(/^error\s+(".*")/) { print OUT $t, "die($1);\n"; } elsif(/^error\s+(.*)/) { print OUT $t, "die(\"", quotemeta($1), "\");\n"; } elsif(/^warning\s+(.*)/) { print OUT $t, "warn(\"", quotemeta($1), "\");\n"; } elsif(/^ident\s+(.*)/) { print OUT $t, "# $1\n"; } } elsif (/^\s*(typedef\s*)?enum\s*(\s+[a-zA-Z_]\w*\s*)?/) { # { for vi until(/\{[^}]*\}.*;/ || /;/) { last unless defined ($next = next_line($file)); chomp $next; # drop "#define FOO FOO" in enums $next =~ s/^\s*#\s*define\s+(\w+)\s+\1\s*$//; # #defines in enums (aliases) $next =~ s/^\s*#\s*define\s+(\w+)\s+(\w+)\s*$/$1 = $2,/; $_ .= $next; print OUT "# $next\n" if $opt_D; } s/#\s*if.*?#\s*endif//g; # drop #ifdefs s@/\*.*?\*/@@g; s/\s+/ /g; next unless /^\s?(typedef\s?)?enum\s?([a-zA-Z_]\w*)?\s?\{(.*)\}\s?([a-zA-Z_]\w*)?\s?;/; (my $enum_subs = $3) =~ s/\s//g; my @enum_subs = split(/,/, $enum_subs); my $enum_val = -1; foreach my $enum (@enum_subs) { my ($enum_name, $enum_value) = $enum =~ /^([a-zA-Z_]\w*)(=.+)?$/; $enum_name or next; $enum_value =~ s/^=//; $enum_val = (length($enum_value) ? $enum_value : $enum_val + 1); if ($opt_h) { print OUT ($t, "eval(\"\\n#line $eval_index $outfile\\n", "sub $enum_name () \{ $enum_val; \}\") ", "unless defined(\&$enum_name);\n"); ++ $eval_index; } else { print OUT ($t, "eval(\"sub $enum_name () \{ $enum_val; \}\") ", "unless defined(\&$enum_name);\n"); } } } elsif (/^(?:__extension__\s+)?(?:extern|static)\s+(?:__)?inline(?:__)?\s+/ and !/;\s*$/ and !/{\s*}\s*$/) { # { for vi # This is a hack to parse the inline functions in the glibc headers. # Warning: massive kludge ahead. We suppose inline functions # are mainly constructed like macros. while (1) { last unless defined ($next = next_line($file)); chomp $next; undef $_, last if $next =~ /__THROW\s*;/ or $next =~ /^(__extension__|extern|static)\b/; $_ .= " $next"; print OUT "# $next\n" if $opt_D; last if $next =~ /^}|^{.*}\s*$/; } next if not defined; # because it's only a prototype s/\b(__extension__|extern|static|(?:__)?inline(?:__)?)\b//g; # violently drop #ifdefs s/#\s*if.*?#\s*endif//g and print OUT "# some #ifdef were dropped here -- fill in the blanks\n"; if (s/^(?:\w|\s|\*)*\s(\w+)\s*//) { $name = $1; } else { warn "name not found"; next; # shouldn't occur... } my @args; if (s/^\(([^()]*)\)\s*(\w+\s*)*//) { for my $arg (split /,/, $1) { if ($arg =~ /(\w+)\s*$/) { $curargs{$1} = 1; push @args, $1; } } } $args = ( @args ? "my(" . (join ',', map "\$$_", @args) . ") = \@_;\n$t " : "" ); my $proto = @args ? '' : '() '; $new = ''; s/\breturn\b//g; # "return" doesn't occur in macros usually... expr(); # try to find and perlify local C variables our @local_variables = (); # needs to be a our(): (?{...}) bug workaround { use re "eval"; my $typelist = join '|', keys %isatype; $new =~ s[' (?:(?:__)?const(?:__)?\s+)? (?:(?:un)?signed\s+)? (?:long\s+)? (?:$typelist)\s+ (\w+) (?{ push @local_variables, $1 }) '] [my \$$1]gx; $new =~ s[' (?:(?:__)?const(?:__)?\s+)? (?:(?:un)?signed\s+)? (?:long\s+)? (?:$typelist)\s+ ' \s+ &(\w+) \s* ; (?{ push @local_variables, $1 }) ] [my \$$1;]gx; } $new =~ s/&$_\b/\$$_/g for @local_variables; $new =~ s/(["\\])/\\$1/g; #"]); # now that's almost like a macro (we hope) EMIT($proto); } } $Is_converted{$file} = 1; if ($opt_e && exists($bad_file{$file})) { unlink($Dest_dir . '/' . $outfile); $next = ''; } else { print OUT "1;\n"; queue_includes_from($file) if $opt_a; } } if ($opt_e && (scalar(keys %bad_file) > 0)) { warn "Was unable to convert the following files:\n"; warn "\t" . join("\n\t",sort(keys %bad_file)) . "\n"; } exit $Exit; sub EMIT { my $proto = shift; $new = reindent($new); $args = reindent($args); if ($t ne '') { $new =~ s/(['\\])/\\$1/g; #']); if ($opt_h) { print OUT $t, "eval \"\\n#line $eval_index $outfile\\n\" . 'sub $name $proto\{\n$t ${args}eval q($new);\n$t}' unless defined(\&$name);\n"; $eval_index++; } else { print OUT $t, "eval 'sub $name $proto\{\n$t ${args}eval q($new);\n$t}' unless defined(\&$name);\n"; } } else { print OUT "unless(defined(\&$name)) {\n sub $name $proto\{\n\t${args}eval q($new);\n }\n}\n"; } %curargs = (); return; } sub expr { if (/\b__asm__\b/) { # freak out $new = '"(assembly code)"'; return } my $joined_args; if(keys(%curargs)) { $joined_args = join('|', keys(%curargs)); } while ($_ ne '') { s/^\&\&// && do { $new .= " &&"; next;}; # handle && operator s/^\&([\(a-z\)]+)/$1/i; # hack for things that take the address of s/^(\s+)// && do {$new .= ' '; next;}; s/^0X([0-9A-F]+)[UL]*//i && do {my $hex = $1; $hex =~ s/^0+//; if (length $hex > 8 && !$Config{use64bitint}) { # Croak if nv_preserves_uv_bits < 64 ? $new .= hex(substr($hex, -8)) + 2**32 * hex(substr($hex, 0, -8)); # The above will produce "erroneous" code # if the hex constant was e.g. inside UINT64_C # macro, but then again, h2ph is an approximation. } else { $new .= lc("0x$hex"); } next;}; s/^(-?\d+\.\d+E[-+]?\d+)[FL]?//i && do {$new .= $1; next;}; s/^(\d+)\s*[LU]*//i && do {$new .= $1; next;}; s/^("(\\"|[^"])*")// && do {$new .= $1; next;}; s/^'((\\"|[^"])*)'// && do { if ($curargs{$1}) { $new .= "ord('\$$1')"; } else { $new .= "ord('$1')"; } next; }; # replace "sizeof(foo)" with "{foo}" # also, remove * (C dereference operator) to avoid perl syntax # problems. Where the %sizeof array comes from is anyone's # guess (c2ph?), but this at least avoids fatal syntax errors. # Behavior is undefined if sizeof() delimiters are unbalanced. # This code was modified to able to handle constructs like this: # sizeof(*(p)), which appear in the HP-UX 10.01 header files. s/^sizeof\s*\(// && do { $new .= '$sizeof'; my $lvl = 1; # already saw one open paren # tack { on the front, and skip it in the loop $_ = "{" . "$_"; my $index = 1; # find balanced closing paren while ($index <= length($_) && $lvl > 0) { $lvl++ if substr($_, $index, 1) eq "("; $lvl-- if substr($_, $index, 1) eq ")"; $index++; } # tack } on the end, replacing ) substr($_, $index - 1, 1) = "}"; # remove pesky * operators within the sizeof argument substr($_, 0, $index - 1) =~ s/\*//g; next; }; # Eliminate typedefs /\(([\w\s]+)[\*\s]*\)\s*[\w\(]/ && do { my $doit = 1; foreach (split /\s+/, $1) { # Make sure all the words are types, unless($isatype{$_} or $_ eq 'struct' or $_ eq 'union'){ $doit = 0; last; } } if( $doit ){ s/\([\w\s]+[\*\s]*\)// && next; # then eliminate them. } }; # struct/union member, including arrays: s/^([_A-Z]\w*(\[[^\]]+\])?((\.|->)[_A-Z]\w*(\[[^\]]+\])?)+)//i && do { my $id = $1; $id =~ s/(\.|(->))([^\.\-]*)/->\{$3\}/g; $id =~ s/\b([^\$])($joined_args)/$1\$$2/g if length($joined_args); while($id =~ /\[\s*([^\$\&\d\]]+)\]/) { my($index) = $1; $index =~ s/\s//g; if(exists($curargs{$index})) { $index = "\$$index"; } else { $index = "&$index"; } $id =~ s/\[\s*([^\$\&\d\]]+)\]/[$index]/; } $new .= " (\$$id)"; }; s/^([_a-zA-Z]\w*)// && do { my $id = $1; if ($id eq 'struct' || $id eq 'union') { s/^\s+(\w+)//; $id .= ' ' . $1; $isatype{$id} = 1; } elsif ($id =~ /^((un)?signed)|(long)|(short)$/) { while (s/^\s+(\w+)//) { $id .= ' ' . $1; } $isatype{$id} = 1; } if ($curargs{$id}) { $new .= "\$$id"; $new .= '->' if /^[\[\{]/; } elsif ($id eq 'defined') { $new .= 'defined'; } elsif (/^\s*\(/) { s/^\s*\((\w),/("$1",/ if $id =~ /^_IO[WR]*$/i; # cheat $new .= " &$id"; } elsif ($isatype{$id}) { if ($new =~ /\{\s*$/) { $new .= "'$id'"; } elsif ($new =~ /\(\s*$/ && /^[\s*]*\)/) { $new =~ s/\(\s*$//; s/^[\s*]*\)//; } else { $new .= q(').$id.q('); } } else { if ($inif) { if ($new =~ /defined\s*$/) { $new .= '(&' . $id . ')'; } elsif ($new =~ /defined\s*\($/) { $new .= '&' . $id; } else { $new .= '(defined(&' . $id . ') ? &' . $id . ' : undef)'; } } elsif (/^\[/) { $new .= " \$$id"; } else { $new .= ' &' . $id; } } next; }; s/^(.)// && do { if ($1 ne '#') { $new .= $1; } next;}; } } sub next_line { my $file = shift; my ($in, $out); my $pre_sub_tri_graphs = 1; READ: while (not eof IN) { $in .= <IN>; chomp $in; next unless length $in; while (length $in) { if ($pre_sub_tri_graphs) { # Preprocess all tri-graphs # including things stuck in quoted string constants. $in =~ s/\?\?=/#/g; # | ??=| #| $in =~ s/\?\?\!/|/g; # | ??!| || $in =~ s/\?\?'/^/g; # | ??'| ^| $in =~ s/\?\?\(/[/g; # | ??(| [| $in =~ s/\?\?\)/]/g; # | ??)| ]| $in =~ s/\?\?\-/~/g; # | ??-| ~| $in =~ s/\?\?\//\\/g; # | ??/| \| $in =~ s/\?\?</{/g; # | ??<| {| $in =~ s/\?\?>/}/g; # | ??>| }| } if ($in =~ /^\#ifdef __LANGUAGE_PASCAL__/) { # Tru64 disassembler.h evilness: mixed C and Pascal. while (<IN>) { last if /^\#endif/; } $in = ""; next READ; } if ($in =~ /^extern inline / && # Inlined assembler. $^O eq 'linux' && $file =~ m!(?:^|/)asm/[^/]+\.h$!) { while (<IN>) { last if /^}/; } $in = ""; next READ; } if ($in =~ s/\\$//) { # \-newline $out .= ' '; next READ; } elsif ($in =~ s/^([^"'\\\/]+)//) { # Passthrough $out .= $1; } elsif ($in =~ s/^(\\.)//) { # \... $out .= $1; } elsif ($in =~ /^'/) { # '... if ($in =~ s/^('(\\.|[^'\\])*')//) { $out .= $1; } else { next READ; } } elsif ($in =~ /^"/) { # "... if ($in =~ s/^("(\\.|[^"\\])*")//) { $out .= $1; } else { next READ; } } elsif ($in =~ s/^\/\/.*//) { # //... # fall through } elsif ($in =~ m/^\/\*/) { # /*... # C comment removal adapted from perlfaq6: if ($in =~ s/^\/\*[^*]*\*+([^\/*][^*]*\*+)*\///) { $out .= ' '; } else { # Incomplete /* */ next READ; } } elsif ($in =~ s/^(\/)//) { # /... $out .= $1; } elsif ($in =~ s/^([^\'\"\\\/]+)//) { $out .= $1; } elsif ($^O eq 'linux' && $file =~ m!(?:^|/)linux/byteorder/pdp_endian\.h$! && $in =~ s!\'T KNOW!!) { $out =~ s!I DON$!I_DO_NOT_KNOW!; } else { if ($opt_e) { warn "Cannot parse $file:\n$in\n"; $bad_file{$file} = 1; $in = ''; $out = undef; last READ; } else { die "Cannot parse:\n$in\n"; } } } last READ if $out =~ /\S/; } return $out; } # Handle recursive subdirectories without getting a grotesquely big stack. # Could this be implemented using File::Find? sub next_file { my $file; while (@ARGV) { $file = shift @ARGV; if ($file eq '-' or -f $file or -l $file) { return $file; } elsif (-d $file) { if ($opt_r) { expand_glob($file); } else { print STDERR "Skipping directory '$file'\n"; } } elsif ($opt_a) { return $file; } else { print STDERR "Skipping '$file': not a file or directory\n"; } } return undef; } # Put all the files in $directory into @ARGV for processing. sub expand_glob { my ($directory) = @_; $directory =~ s:/$::; opendir DIR, $directory; foreach (readdir DIR) { next if ($_ eq '.' or $_ eq '..'); # expand_glob() is going to be called until $ARGV[0] isn't a # directory; so push directories, and unshift everything else. if (-d "$directory/$_") { push @ARGV, "$directory/$_" } else { unshift @ARGV, "$directory/$_" } } closedir DIR; } # Given $file, a symbolic link to a directory in the C include directory, # make an equivalent symbolic link in $Dest_dir, if we can figure out how. # Otherwise, just duplicate the file or directory. sub link_if_possible { my ($dirlink) = @_; my $target = eval 'readlink($dirlink)'; if ($target =~ m:^\.\./: or $target =~ m:^/:) { # The target of a parent or absolute link could leave the $Dest_dir # hierarchy, so let's put all of the contents of $dirlink (actually, # the contents of $target) into @ARGV; as a side effect down the # line, $dirlink will get created as an _actual_ directory. expand_glob($dirlink); } else { if (-l "$Dest_dir/$dirlink") { unlink "$Dest_dir/$dirlink" or print STDERR "Could not remove link $Dest_dir/$dirlink: $!\n"; } if (eval 'symlink($target, "$Dest_dir/$dirlink")') { print "Linking $target -> $Dest_dir/$dirlink\n"; # Make sure that the link _links_ to something: if (! -e "$Dest_dir/$target") { mkpath("$Dest_dir/$target", 0755) or print STDERR "Could not create $Dest_dir/$target/\n"; } } else { print STDERR "Could not symlink $target -> $Dest_dir/$dirlink: $!\n"; } } } # Push all #included files in $file onto our stack, except for STDIN # and files we've already processed. sub queue_includes_from { my ($file) = @_; my $line; return if ($file eq "-"); open HEADER, "<", $file or return; while (defined($line = <HEADER>)) { while (/\\$/) { # Handle continuation lines chop $line; $line .= <HEADER>; } if ($line =~ /^#\s*include\s+([<"])(.*?)[>"]/) { my ($delimiter, $new_file) = ($1, $2); # copy the prefix in the quote syntax (#include "x.h") case if ($delimiter eq q{"} && $file =~ m|^(.*)/|) { $new_file = "$1/$new_file"; } push(@ARGV, $new_file) unless $Is_converted{$new_file}; } } close HEADER; } # Determine include directories; $Config{usrinc} should be enough for (all # non-GCC?) C compilers, but gcc uses additional include directories. sub inc_dirs { my $from_gcc = `LC_ALL=C $Config{cc} -v -E - < /dev/null 2>&1 | awk '/^#include/, /^End of search list/' | grep '^ '`; length($from_gcc) ? (split(' ', $from_gcc), $Config{usrinc}) : ($Config{usrinc}); } # Create "_h2ph_pre.ph", if it doesn't exist or was built by a different # version of h2ph. sub build_preamble_if_necessary { # Increment $VERSION every time this function is modified: my $VERSION = 4; my $preamble = "$Dest_dir/_h2ph_pre.ph"; # Can we skip building the preamble file? if (-r $preamble) { # Extract version number from first line of preamble: open PREAMBLE, "<", $preamble or die "Cannot open $preamble: $!"; my $line = <PREAMBLE>; $line =~ /(\b\d+\b)/; close PREAMBLE or die "Cannot close $preamble: $!"; # Don't build preamble if a compatible preamble exists: return if $1 == $VERSION; } my (%define) = _extract_cc_defines(); open PREAMBLE, ">", $preamble or die "Cannot open $preamble: $!"; print PREAMBLE "# This file was created by h2ph version $VERSION\n"; # Prevent non-portable hex constants from warning. # # We still produce an overflow warning if we can't represent # a hex constant as an integer. print PREAMBLE "no warnings qw(portable);\n"; foreach (sort keys %define) { if ($opt_D) { print PREAMBLE "# $_=$define{$_}\n"; } if ($define{$_} =~ /^\((.*)\)$/) { # parenthesized value: d=(v) $define{$_} = $1; } if (/^(\w+)\((\w)\)$/) { my($macro, $arg) = ($1, $2); my $def = $define{$_}; $def =~ s/$arg/\$\{$arg\}/g; print PREAMBLE <<DEFINE; unless (defined &$macro) { sub $macro(\$) { my (\$$arg) = \@_; \"$def\" } } DEFINE } elsif ($define{$_} =~ /^([+-]?(\d+)?\.\d+([eE][+-]?\d+)?)[FL]?$/) { # float: print PREAMBLE "unless (defined &$_) { sub $_() { $1 } }\n\n"; } elsif ($define{$_} =~ /^([+-]?\d+)U?L{0,2}$/i) { # integer: print PREAMBLE "unless (defined &$_) { sub $_() { $1 } }\n\n"; } elsif ($define{$_} =~ /^([+-]?0x[\da-f]+)U?L{0,2}$/i) { # hex integer # Special cased, since perl warns on hex integers # that can't be represented in a UV. # # This way we get the warning at time of use, so the user # only gets the warning if they happen to use this # platform-specific definition. my $code = $1; $code = "hex('$code')" if length $code > 10; print PREAMBLE "unless (defined &$_) { sub $_() { $code } }\n\n"; } elsif ($define{$_} =~ /^\w+$/) { my $def = $define{$_}; if ($isatype{$def}) { print PREAMBLE "unless (defined &$_) { sub $_() { \"$def\" } }\n\n"; } else { print PREAMBLE "unless (defined &$_) { sub $_() { &$def } }\n\n"; } } else { print PREAMBLE "unless (defined &$_) { sub $_() { \"", quotemeta($define{$_}), "\" } }\n\n"; } } print PREAMBLE "\n1;\n"; # avoid 'did not return a true value' when empty close PREAMBLE or die "Cannot close $preamble: $!"; } # %Config contains information on macros that are pre-defined by the # system's compiler. We need this information to make the .ph files # function with perl as the .h files do with cc. sub _extract_cc_defines { my %define; my $allsymbols = join " ", @Config{'ccsymbols', 'cppsymbols', 'cppccsymbols'}; # If optimizing -O2 is used, add the definition if ($Config{'ccflags'} =~ /(?:\s+|^)-O([\d]+)(?:\s+|$)/) { $allsymbols .= " __OPTIMIZE__=$1"; } # Split compiler pre-definitions into 'key=value' pairs: while ($allsymbols =~ /([^\s]+)=((\\\s|[^\s])+)/g) { $define{$1} = $2; if ($opt_D) { print STDERR "$_: $1 -> $2\n"; } } return %define; } 1; ############################################################################## __END__ =head1 NAME h2ph - convert .h C header files to .ph Perl header files =head1 SYNOPSIS B<h2ph [-d destination directory] [-r | -a] [-l] [-h] [-e] [-D] [-Q] [headerfiles]> =head1 DESCRIPTION I<h2ph> converts any C header files specified to the corresponding Perl header file format. It is most easily run while in /usr/include: cd /usr/include; h2ph * sys/* or cd /usr/include; h2ph * sys/* arpa/* netinet/* or cd /usr/include; h2ph -r -l . The output files are placed in the hierarchy rooted at Perl's architecture dependent library directory. You can specify a different hierarchy with a B<-d> switch. If run with no arguments, filters standard input to standard output. =head1 OPTIONS =over 4 =item -d destination_dir Put the resulting B<.ph> files beneath B<destination_dir>, instead of beneath the default Perl library location (C<$Config{'installsitearch'}>). =item -r Run recursively; if any of B<headerfiles> are directories, then run I<h2ph> on all files in those directories (and their subdirectories, etc.). B<-r> and B<-a> are mutually exclusive. =item -a Run automagically; convert B<headerfiles>, as well as any B<.h> files which they include. This option will search for B<.h> files in all directories which your C compiler ordinarily uses. B<-a> and B<-r> are mutually exclusive. =item -l Symbolic links will be replicated in the destination directory. If B<-l> is not specified, then links are skipped over. =item -h Put 'hints' in the .ph files which will help in locating problems with I<h2ph>. In those cases when you B<require> a B<.ph> file containing syntax errors, instead of the cryptic [ some error condition ] at (eval mmm) line nnn you will see the slightly more helpful [ some error condition ] at filename.ph line nnn However, the B<.ph> files almost double in size when built using B<-h>. =item -e If an error is encountered during conversion, output file will be removed and a warning emitted instead of terminating the conversion immediately. =item -D Include the code from the B<.h> file as a comment in the B<.ph> file. This is primarily used for debugging I<h2ph>. =item -Q 'Quiet' mode; don't print out the names of the files being converted. =back =head1 ENVIRONMENT No environment variables are used. =head1 FILES /usr/include/*.h /usr/include/sys/*.h etc. =head1 AUTHOR Larry Wall =head1 SEE ALSO perl(1) =head1 DIAGNOSTICS The usual warnings if it can't read or write the files involved. =head1 BUGS Doesn't construct the %sizeof array for you. It doesn't handle all C constructs, but it does attempt to isolate definitions inside evals so that you can get at the definitions that it can translate. It's only intended as a rough tool. You may need to dicker with the files produced. You have to run this program by hand; it's not run as part of the Perl installation. Doesn't handle complicated expressions built piecemeal, a la: enum { FIRST_VALUE, SECOND_VALUE, #ifdef ABC THIRD_VALUE #endif }; Doesn't necessarily locate all of your C compiler's internally-defined symbols. =cut
Save