my
$generation
= 0;
sub
TIEHASH
{
my
(
$class
,
$rule
) =
@_
;
return
bless
\
$rule
,
$class
;
}
sub
FETCH
{
my
$self
=
shift
;
local
$_
=
shift
;
my
$rule
=
$$self
;
return
undef
unless
(/^[\@^<?*]$/);
return
$rule
->Name
if
(
$_
eq
'@'
);
return
$rule
->Base
if
(
$_
eq
'*'
);
return
join
(
' '
,
$rule
->exp_depend)
if
(
$_
eq
'^'
);
return
join
(
' '
,
$rule
->out_of_date)
if
(
$_
eq
'?'
);
return
(
$rule
->exp_depend)[0]
if
(
$_
eq
'<'
);
return
undef
;
}
sub
target
{
return
shift
->{TARGET};
}
sub
Name
{
return
shift
->target->Name;
}
sub
Base
{
my
$name
=
shift
->target->Name;
$name
=~ s/\.[^.]+$//;
return
$name
;
}
sub
Info
{
return
shift
->target->Info;
}
sub
depend
{
my
$self
=
shift
;
if
(
@_
)
{
my
$name
=
$self
->Name;
my
$dep
=
shift
;
confess
"dependants $dep are not an array reference"
unless
(
'ARRAY'
eq
ref
$dep
);
my
$file
;
foreach
$file
(
@$dep
)
{
unless
(
exists
$self
->{DEPHASH}{
$file
})
{
$self
->{DEPHASH}{
$file
} = 1;
push
(@{
$self
->{DEPEND}},
$file
);
}
}
}
return
(
wantarray
) ? @{
$self
->{DEPEND}} :
$self
->{DEPEND};
}
sub
command
{
my
$self
=
shift
;
if
(
@_
)
{
my
$cmd
=
shift
;
confess
"commands $cmd are not an array reference"
unless
(
'ARRAY'
eq
ref
$cmd
);
if
(
@$cmd
)
{
if
(@{
$self
->{COMMAND}})
{
warn
"Command for "
.
$self
->Name,
" redefined"
;
print
STDERR
"Was:"
,
join
(
"\n"
,@{
$self
->{COMMAND}}),
"\n"
;
print
STDERR
"Now:"
,
join
(
"\n"
,
@$cmd
),
"\n"
;
}
$self
->{COMMAND} =
$cmd
;
}
else
{
if
(@{
$self
->{COMMAND}})
{
}
}
}
return
(
wantarray
) ? @{
$self
->{COMMAND}} :
$self
->{COMMAND};
}
sub
out_of_date
{
my
$array
=
wantarray
;
my
$self
=
shift
;
my
$info
=
$self
->Info;
my
@dep
= ();
my
$tdate
=
$self
->target->date;
my
$dep
;
my
$count
= 0;
foreach
$dep
(
$self
->exp_depend)
{
my
$date
=
$info
->date(
$dep
);
$count
++;
if
(!
defined
(
$date
) || !
defined
(
$tdate
) ||
$date
<
$tdate
)
{
return
1
unless
$array
;
push
(
@dep
,
$dep
);
}
}
return
@dep
if
$array
;
return
!
$count
;
}
sub
exp_depend
{
my
$self
=
shift
;
my
$info
=
$self
->Info;
my
@dep
=
map
(
split
(/\s+/,
$info
->subsvars(
$_
)),
$self
->depend);
return
(
wantarray
) ?
@dep
: \
@dep
;
}
sub
exp_command
{
my
$self
=
shift
;
my
$info
=
$self
->Info;
my
$base
=
$self
->Name;
my
%var
;
tie
%var
,
'Make::Rule::Vars'
,
$self
;
my
@cmd
=
map
(
$info
->subsvars(
$_
,\
%var
),
$self
->command);
return
(
wantarray
) ?
@cmd
: \
@cmd
;
}
sub
clone
{
my
(
$self
,
$target
) =
@_
;
my
%hash
=
%$self
;
$hash
{TARGET} =
$target
;
$hash
{DEPEND} = [@{
$self
->{DEPEND}}];
$hash
{DEPHASH} = {%{
$self
->{DEPHASH}}};
my
$obj
=
bless
\
%hash
,
ref
$self
;
return
$obj
;
}
sub
new
{
my
$class
=
shift
;
my
$target
=
shift
;
my
$kind
=
shift
;
my
$self
=
bless
{
TARGET
=>
$target
,
KIND
=>
$kind
,
DEPEND
=> [],
DEPHASH
=> {},
COMMAND
=> []
},
$class
;
$self
->depend(
shift
)
if
(
@_
);
$self
->command(
shift
)
if
(
@_
);
return
$self
;
}
sub
find_commands
{
my
(
$self
) =
@_
;
if
(!@{
$self
->{COMMAND}} && @{
$self
->{DEPEND}})
{
my
$info
=
$self
->Info;
my
$name
=
$self
->Name;
my
@dep
=
$self
->depend;
my
@rule
=
$info
->patrule(
$self
->Name);
if
(
@rule
)
{
$self
->depend(
$rule
[0]);
$self
->command(
$rule
[1]);
}
}
}
sub
Script
{
my
$self
=
shift
;
return
unless
$self
->out_of_date;
my
@cmd
=
$self
->exp_command;
if
(
@cmd
)
{
my
$file
;
my
$com
= ($^O eq
'MSWin32'
) ?
'rem '
:
'# '
;
print
$com
,
$self
->Name,
"\n"
;
foreach
$file
(
$self
->exp_command)
{
$file
=~ s/^[\@\s-]*//;
print
"$file\n"
;
}
}
}
sub
Make
{
my
$self
=
shift
;
my
$file
;
return
unless
(
$self
->out_of_date);
my
@cmd
=
$self
->exp_command;
my
$info
=
$self
->Info;
if
(
@cmd
)
{
foreach
my
$file
(
$self
->exp_command)
{
$file
=~ s/^([\@\s-]*)//;
my
$prefix
= $1;
print
"$file\n"
unless
(
$prefix
=~ /\@/);
my
$code
=
$info
->
exec
(
$file
);
if
(
$code
&&
$prefix
!~ /-/)
{
die
"Code $code from $file"
;
}
}
}
}
sub
Print
{
my
$self
=
shift
;
my
$file
;
print
$self
->Name,
' '
,
$self
->{KIND},
' '
;
foreach
$file
(
$self
->depend)
{
print
" \\\n $file"
;
}
print
"\n"
;
my
@cmd
=
$self
->exp_command;
if
(
@cmd
)
{
foreach
$file
(
$self
->exp_command)
{
print
"\t"
,
$file
,
"\n"
;
}
}
else
{
print
STDERR
"No commands for "
,
$self
->Name,
"\n"
unless
(
$self
->target->phony);
}
print
"\n"
;
}
sub
new
{
my
(
$class
,
$info
,
$target
) =
@_
;
return
bless
{
NAME
=>
$target
,
MAKEFILE
=>
$info
,
Pass
=> 0
},
$class
;
}
sub
date
{
my
$self
=
shift
;
my
$info
=
$self
->Info;
return
$info
->date(
$self
->Name);
}
sub
phony
{
my
$self
=
shift
;
return
$self
->Info->phony(
$self
->Name);
}
sub
colon
{
my
$self
=
shift
;
if
(
@_
)
{
if
(
exists
$self
->{COLON})
{
my
$dep
=
$self
->{COLON};
if
(
@_
== 1)
{
my
$other
=
shift
;
$dep
->depend(
scalar
$other
->depend);
$dep
->command(
scalar
$other
->command);
}
else
{
$dep
->depend(
shift
);
$dep
->command(
shift
);
}
}
else
{
$self
->{COLON} = (
@_
== 1) ?
shift
->clone(
$self
) : Make::Rule->new(
$self
,
':'
,
@_
);
}
}
if
(
exists
$self
->{COLON})
{
return
(
wantarray
) ? (
$self
->{COLON}) :
$self
->{COLON};
}
else
{
return
(
wantarray
) ? () :
undef
;
}
}
sub
dcolon
{
my
$self
=
shift
;
if
(
@_
)
{
my
$rule
= (
@_
== 1) ?
shift
->clone(
$self
) : Make::Rule->new(
$self
,
'::'
,
@_
);
$self
->{DCOLON} = []
unless
(
exists
$self
->{DCOLON});
push
(@{
$self
->{DCOLON}},
$rule
);
}
return
(
exists
$self
->{DCOLON}) ? @{
$self
->{DCOLON}} : ();
}
sub
Name
{
return
shift
->{NAME};
}
sub
Info
{
return
shift
->{MAKEFILE};
}
sub
ProcessColon
{
my
(
$self
) =
@_
;
my
$c
=
$self
->colon;
$c
->find_commands
if
$c
;
}
sub
ExpandTarget
{
my
(
$self
) =
@_
;
my
$target
=
$self
->Name;
my
$info
=
$self
->Info;
my
$colon
=
delete
$self
->{COLON};
my
$dcolon
=
delete
$self
->{DCOLON};
foreach
my
$expand
(
split
(/\s+/,
$info
->subsvars(
$target
)))
{
next
unless
defined
(
$expand
);
my
$t
=
$info
->Target(
$expand
);
if
(
defined
$colon
)
{
$t
->colon(
$colon
);
}
foreach
my
$d
(@{
$dcolon
})
{
$t
->dcolon(
$d
);
}
}
}
sub
done
{
my
$self
=
shift
;
my
$info
=
$self
->Info;
my
$pass
=
$info
->pass;
return
1
if
(
$self
->{Pass} ==
$pass
);
$self
->{Pass} =
$pass
;
return
0;
}
sub
recurse
{
my
(
$self
,
$method
,
@args
) =
@_
;
my
$info
=
$self
->Info;
my
$rule
;
my
$i
= 0;
foreach
$rule
(
$self
->colon,
$self
->dcolon)
{
my
$dep
;
my
$j
= 0;
foreach
$dep
(
$rule
->exp_depend)
{
my
$t
=
$info
->{Depend}{
$dep
};
if
(
defined
$t
)
{
$t
->
$method
(
@args
)
}
else
{
unless
(
$info
->
exists
(
$dep
))
{
my
$dir
= cwd();
die
"Cannot recurse $method - no target $dep in $dir"
}
}
}
}
}
sub
Script
{
my
$self
=
shift
;
my
$info
=
$self
->Info;
my
$rule
=
$self
->colon;
return
if
(
$self
->done);
$self
->recurse(
'Script'
);
foreach
$rule
(
$self
->colon,
$self
->dcolon)
{
$rule
->Script;
}
}
sub
Make
{
my
$self
=
shift
;
my
$info
=
$self
->Info;
my
$rule
=
$self
->colon;
return
if
(
$self
->done);
$self
->recurse(
'Make'
);
foreach
$rule
(
$self
->colon,
$self
->dcolon)
{
$rule
->Make;
}
}
sub
Print
{
my
$self
=
shift
;
my
$info
=
$self
->Info;
return
if
(
$self
->done);
my
$rule
=
$self
->colon;
foreach
$rule
(
$self
->colon,
$self
->dcolon)
{
$rule
->Print;
}
$self
->recurse(
'Print'
);
}
use
5.005;
$VERSION
=
'1.00'
;
my
%date
;
sub
phony
{
my
(
$self
,
$name
) =
@_
;
return
exists
$self
->{PHONY}{
$name
};
}
sub
suffixes
{
my
(
$self
) =
@_
;
return
keys
%{
$self
->{
'SUFFIXES'
}};
}
sub
Target
{
my
(
$self
,
$target
) =
@_
;
unless
(
exists
$self
->{Depend}{
$target
})
{
my
$t
= Make::Target->new(
$self
,
$target
);
$self
->{Depend}{
$target
} =
$t
;
if
(
$target
=~ /%/)
{
$self
->{Pattern}{
$target
} =
$t
;
}
elsif
(
$target
=~ /^\./)
{
$self
->{Dot}{
$target
} =
$t
;
}
else
{
push
(@{
$self
->{Targets}},
$t
);
}
}
return
$self
->{Depend}{
$target
};
}
sub
patmatch
{
my
$key
=
shift
;
local
$_
=
shift
;
my
$pat
=
$key
;
$pat
=~ s/\./\\./;
$pat
=~ s/%/(\[^\/\]*)/;
if
(/
$pat
$/)
{
return
$1;
}
return
undef
;
}
sub
locate
{
my
$self
=
shift
;
local
$_
=
shift
;
return
$_
if
(-r
$_
);
my
$key
;
foreach
$key
(
keys
%{
$self
->{vpath}})
{
my
$Pat
;
if
(
defined
(
$Pat
= patmatch(
$key
,
$_
)))
{
my
$dir
;
foreach
$dir
(
split
(/:/,
$self
->{vpath}{
$key
}))
{
return
"$dir/$_"
if
(-r
"$dir/$_"
);
}
}
}
return
undef
;
}
sub
dotrules
{
my
(
$self
) =
@_
;
my
$t
;
foreach
$t
(
keys
%{
$self
->{Dot}})
{
my
$e
=
$self
->subsvars(
$t
);
$self
->{Dot}{
$e
} =
delete
$self
->{Dot}{
$t
}
unless
(
$t
eq
$e
);
}
my
(
@suffix
) =
$self
->suffixes;
foreach
$t
(
@suffix
)
{
my
$d
;
my
$r
=
delete
$self
->{Dot}{
$t
};
if
(
defined
$r
)
{
my
@rule
= (
$r
->colon) ? (
$r
->colon->depend) : ();
if
(
@rule
)
{
delete
$self
->{Dot}{
$t
->Name};
print
STDERR
$t
->Name,
" has dependants\n"
;
push
(@{
$self
->{Targets}},
$r
);
}
else
{
$self
->Target(
'%'
)->dcolon([
'%'
.
$t
],
scalar
$r
->colon->command);
}
}
foreach
$d
(
@suffix
)
{
$r
=
delete
$self
->{Dot}{
$t
.
$d
};
if
(
defined
$r
)
{
$self
->Target(
'%'
.
$d
)->dcolon([
'%'
.
$t
],
scalar
$r
->colon->command);
}
}
}
foreach
$t
(
keys
%{
$self
->{Dot}})
{
push
(@{
$self
->{Targets}},
delete
$self
->{Dot}{
$t
});
}
}
my
%pathname
;
sub
pathname
{
my
(
$self
,
$name
) =
@_
;
my
$hash
=
$self
->{
'Pathname'
};
unless
(
exists
$hash
->{
$name
})
{
if
(File::Spec->file_name_is_absolute(
$name
))
{
$hash
->{
$name
} =
$name
;
}
else
{
$name
=~ s,^\./,,;
$hash
->{
$name
} = File::Spec->catfile(
$self
->{Dir},
$name
);
}
}
return
$hash
->{
$name
};
}
sub
date
{
my
(
$self
,
$name
) =
@_
;
my
$path
=
$self
->pathname(
$name
);
unless
(
exists
$date
{
$path
})
{
$date
{
$path
} = -M
$path
;
}
return
$date
{
$path
};
}
sub
exists
{
my
(
$self
,
$name
) =
@_
;
return
1
if
(
exists
$self
->{Depend}{
$name
});
return
1
if
defined
$self
->date(
$name
);
return
0;
}
sub
patrule
{
my
(
$self
,
$target
) =
@_
;
my
$key
;
foreach
$key
(
keys
%{
$self
->{Pattern}})
{
my
$Pat
;
if
(
defined
(
$Pat
= patmatch(
$key
,
$target
)))
{
my
$t
=
$self
->{Pattern}{
$key
};
my
$rule
;
foreach
$rule
(
$t
->dcolon)
{
my
@dep
=
$rule
->exp_depend;
if
(
@dep
)
{
my
$dep
=
$dep
[0];
$dep
=~ s/%/
$Pat
/g;
if
(
$self
->
exists
(
$dep
))
{
foreach
(
@dep
)
{
s/%/
$Pat
/g;
}
return
(\
@dep
,
scalar
$rule
->command);
}
}
}
}
}
return
();
}
sub
needs
{
my
(
$self
,
$target
) =
@_
;
unless
(
$self
->{Done}{
$target
})
{
if
(
exists
$self
->{Depend}{
$target
})
{
my
@depend
=
split
(/\s+/,
$self
->subsvars(
$self
->{Depend}{
$target
}));
foreach
(
@depend
)
{
$self
->needs(
$_
);
}
}
else
{
my
$vtarget
=
$self
->locate(
$target
);
if
(
defined
$vtarget
)
{
$self
->{Need}{
$vtarget
} =
$target
;
}
else
{
$self
->{Need}{
$target
} =
$target
;
}
}
}
}
sub
subsvars
{
my
$self
=
shift
;
local
$_
=
shift
;
my
@var
=
@_
;
push
(
@var
,
$self
->{Override},
$self
->{Vars},\
%ENV
);
croak(
"Trying to subsitute undef value"
)
unless
(
defined
$_
);
while
(/(?<!\$)\$\(([^()]+)\)/ || /(?<!\$)\$([<\@^?*])/)
{
my
(
$key
,
$head
,
$tail
) = ($1,$`,$');
my
$value
;
if
(
$key
=~ /^([\w._]+|\S)(?::(.*))?$/)
{
my
(
$var
,
$op
) = ($1,$2);
foreach
my
$hash
(
@var
)
{
$value
=
$hash
->{
$var
};
if
(
defined
$value
)
{
last
;
}
}
unless
(
defined
$value
)
{
die
"$var not defined in '$_'"
unless
(
length
(
$var
) > 1);
$value
=
''
;
}
if
(
defined
$op
)
{
if
(
$op
=~ /^s(.).*\1.*\1/)
{
local
$_
=
$self
->subsvars(
$value
);
$op
=~ s/\\/\\\\/g;
eval
$op
.
'g'
;
$value
=
$_
;
}
else
{
die
"$var:$op = '$value'\n"
;
}
}
}
elsif
(
$key
=~ /wildcard\s*(.*)$/)
{
$value
=
join
(
' '
,
glob
(
$self
->pathname($1)));
}
elsif
(
$key
=~ /shell\s*(.*)$/)
{
$value
=
join
(
' '
,
split
(
'\n'
,`$1`));
}
elsif
(
$key
=~ /addprefix\s*([^,]*),(.*)$/)
{
$value
=
join
(
' '
,
map
($1 .
$_
,
split
(
'\s+'
,$2)));
}
elsif
(
$key
=~ /notdir\s*(.*)$/)
{
my
@files
=
split
(/\s+/,$1);
foreach
(
@files
)
{
s
}
$value
=
join
(
' '
,
@files
);
}
elsif
(
$key
=~ /dir\s*(.*)$/)
{
my
@files
=
split
(/\s+/,$1);
foreach
(
@files
)
{
s
}
$value
=
join
(
' '
,
@files
);
}
elsif
(
$key
=~ /^subst\s+([^,]*),([^,]*),(.*)$/)
{
my
(
$a
,
$b
) = ($1,$2);
$value
= $3;
$a
=~ s/\./\\./;
$value
=~ s/
$a
/
$b
/;
}
elsif
(
$key
=~ /^mktmp,(\S+)\s*(.*)$/)
{
my
(
$file
,
$content
) = ($1,$2);
open
(TMP,
">$file"
) ||
die
"Cannot open $file:$!"
;
$content
=~ s/\\n//g;
print
TMP
$content
;
close
(TMP);
$value
=
$file
;
}
else
{
warn
"Cannot evaluate '$key' in '$_'\n"
;
}
$_
=
"$head$value$tail"
;
}
s/\$\$/\$/g;
return
$_
;
}
sub
tokenize
{
local
$_
=
$_
[0];
my
@result
= ();
s/\s+$//;
while
(
length
(
$_
))
{
s/^\s+//;
last
unless
(/^\S/);
my
$token
=
""
;
while
(/^\S/)
{
if
(s/^\$([\(\{])//)
{
$token
.= $&;
my
$paren
= $1 eq
'('
;
my
$brace
= $1 eq
'{'
;
my
$count
= 1;
while
(
length
(
$_
) && (
$paren
||
$brace
))
{
s/^.//;
$token
.= $&;
$paren
+= ($& eq
'('
);
$paren
-= ($& eq
')'
);
$brace
+= ($& eq
'{'
);
$brace
-= ($& eq
'}'
);
}
die
"Mismatched {} in $_[0]"
if
(
$brace
);
die
"Mismatched () in $_[0]"
if
(
$paren
);
}
elsif
(s/^(\$\S?|[^\s\$]+)//)
{
$token
.= $&;
}
}
push
(
@result
,
$token
);
}
return
(
wantarray
) ?
@result
: \
@result
;
}
sub
makefile
{
my
(
$self
,
$makefile
,
$name
) =
@_
;
local
$_
;
print
STDERR
"Reading $name\n"
;
Makefile:
while
(<
$makefile
>)
{
last
unless
(
defined
$_
);
chomp
(
$_
);
if
(/\\$/)
{
chop
(
$_
);
s/\s*$//;
my
$more
= <
$makefile
>;
$more
=~ s/^\s*/ /;
$_
.=
$more
;
redo
;
}
next
if
(/^\s*
next
if
(/^\s*$/);
s/
s/^\s+//;
if
(/^(-?)include\s+(.*)$/)
{
my
$opt
= $1;
my
$file
;
foreach
$file
(tokenize(
$self
->subsvars($2)))
{
local
*Makefile
;
my
$path
=
$self
->pathname(
$file
);
if
(
open
(Makefile,
"<$path"
))
{
$self
->makefile(\
*Makefile
,
$path
);
close
(Makefile);
}
else
{
warn
"Cannot open $path:$!"
unless
(
$opt
eq
'-'
) ;
}
}
}
elsif
(/^\s*([\w._]+)\s*:?=\s*(.*)$/)
{
$self
->{Vars}{$1} = (
defined
$2) ? $2 :
""
;
}
elsif
(/^vpath\s+(\S+)\s+(.*)$/)
{
my
(
$pat
,
$path
) = ($1,$2);
$self
->{Vpath}{
$pat
} =
$path
;
}
elsif
(/^\s*([^:]*)(::?)\s*(.*)$/)
{
my
(
$target
,
$kind
,
$depend
) = ($1,$2,$3);
my
@cmnds
;
if
(
$depend
=~ /^([^;]*);(.*)$/)
{
(
$depend
,
$cmnds
[0]) = ($1,$2);
}
while
(<
$makefile
>)
{
next
if
(/^\s*
next
if
(/^\s*$/);
last
unless
(/^\t/);
chop
(
$_
);
if
(/\\$/)
{
chop
(
$_
);
$_
.=
' '
;
$_
.= <
$makefile
>;
redo
;
}
next
if
(/^\s*$/);
s/^\s+//;
push
(
@cmnds
,
$_
);
}
$depend
=~ s/\s\s+/ /;
$target
=~ s/\s\s+/ /;
my
@depend
= tokenize(
$depend
);
foreach
(tokenize(
$target
))
{
my
$t
=
$self
->Target(
$_
);
my
$index
= 0;
if
(
$kind
eq
'::'
|| /%/)
{
$t
->dcolon(\
@depend
,\
@cmnds
);
}
else
{
$t
->colon(\
@depend
,\
@cmnds
);
}
}
redo
Makefile;
}
else
{
warn
"Ignore '$_'\n"
;
}
}
}
sub
pseudos
{
my
$self
=
shift
;
my
$key
;
foreach
$key
(
qw(SUFFIXES PHONY PRECIOUS PARALLEL)
)
{
my
$t
=
delete
$self
->{Dot}{
'.'
.
$key
};
if
(
defined
$t
)
{
my
$dep
;
$self
->{
$key
} = {};
foreach
$dep
(
$t
->colon->exp_depend)
{
$self
->{
$key
}{
$dep
} = 1;
}
}
}
}
sub
ExpandTarget
{
my
$self
=
shift
;
foreach
my
$t
(@{
$self
->{
'Targets'
}})
{
$t
->ExpandTarget;
}
foreach
my
$t
(@{
$self
->{
'Targets'
}})
{
$t
->ProcessColon;
}
}
sub
parse
{
my
(
$self
,
$file
) =
@_
;
if
(
defined
$file
)
{
$file
=
$self
->pathname(
$file
);
}
else
{
my
@files
=
qw(makefile Makefile)
;
unshift
(
@files
,
'GNUmakefile'
)
if
(
$self
->{GNU});
my
$name
;
foreach
$name
(
@files
)
{
$file
=
$self
->pathname(
$name
);
if
(-r
$file
)
{
$self
->{Makefile} =
$name
;
last
;
}
}
}
local
(
*Makefile
);
open
(Makefile,
"<$file"
) || croak(
"Cannot open $file:$!"
);
$self
->makefile(\
*Makefile
,
$file
);
close
(Makefile);
$self
->pseudos;
$self
->dotrules;
}
sub
PrintVars
{
my
$self
=
shift
;
local
$_
;
foreach
(
keys
%{
$self
->{Vars}})
{
print
"$_ = "
,
$self
->{Vars}{
$_
},
"\n"
;
}
print
"\n"
;
}
sub
exec
{
my
$self
=
shift
;
undef
%date
;
$generation
++;
if
($^O eq
'MSWin32'
)
{
my
$cwd
= cwd();
my
$ret
;
chdir
$self
->{Dir};
$ret
=
system
(
@_
);
chdir
$cwd
;
return
$ret
;
}
else
{
my
$pid
=
fork
;
if
(
$pid
)
{
waitpid
$pid
,0;
return
$?;
}
else
{
my
$dir
=
$self
->{Dir};
chdir
(
$dir
) ||
die
"Cannot cd to $dir"
;
exec
(
"/bin/sh"
,
"-c"
,
@_
) || confess
"Cannot exec "
.
join
(
' '
,
@_
);
}
}
}
sub
NextPass {
shift
->{Pass}++ }
sub
pass {
shift
->{Pass} }
sub
apply
{
my
$self
=
shift
;
my
$method
=
shift
;
$self
->NextPass;
my
@targets
= ();
foreach
(
@_
)
{
if
(/^(\w+)=(.*)$/)
{
$self
->{Override}{$1} = $2;
}
else
{
push
(
@targets
,
$_
);
}
}
$self
->ExpandTarget;
@targets
= (
$self
->{
'Targets'
}[0])->Name
unless
(
@targets
);
foreach
(
@targets
)
{
my
$t
=
$self
->{Depend}{
$_
};
unless
(
defined
$t
)
{
print
STDERR
join
(
' '
,
$method
,
@_
),
"\n"
;
die
"Cannot `$method' - no target $_"
}
$t
->
$method
();
}
}
sub
Script
{
shift
->apply(
Script
=>
@_
);
}
sub
Print
{
shift
->apply(
Print
=>
@_
);
}
sub
Make
{
shift
->apply(
Make
=>
@_
);
}
sub
new
{
my
(
$class
,
%args
) =
@_
;
unless
(
defined
$args
{Dir})
{
chomp
(
$args
{Dir} = getcwd());
}
my
$self
=
bless
{
%args
,
Pattern
=> {},
Dot
=> {},
Vpath
=> {},
Vars
=> {},
Depend
=> {},
Targets
=> [],
Pass
=> 0,
Pathname
=> {},
Need
=> {},
Done
=> {},
},
$class
;
$self
->{Vars}{CC} =
$Config
{cc};
$self
->{Vars}{AR} =
$Config
{ar};
$self
->{Vars}{CFLAGS} =
$Config
{optimize};
$self
->makefile(\
*DATA
,__FILE__);
$self
->parse(
$self
->{Makefile});
return
$self
;
}
1;