ViewVC Help
View File | Revision Log | Show Annotations | Root Listing
root/cvsroot/COMP/SCRAM/src/BuildSystem/BuildFile.pm
(Generate patch)

Comparing COMP/SCRAM/src/BuildSystem/BuildFile.pm (file contents):
Revision 1.12 by williamc, Tue Nov 14 15:18:41 2000 UTC vs.
Revision 1.30 by sashby, Tue Feb 27 11:59:44 2007 UTC

# Line 1 | Line 1
1 < # BuildFile
1 > #____________________________________________________________________
2 > # File: BuildFile.pm
3 > #____________________________________________________________________
4 > #  
5 > # Author: Shaun Ashby <Shaun.Ashby@cern.ch>
6 > # Update: 2003-12-03 19:03:15+0100
7 > # Revision: $Id$
8   #
9 < # Interface
10 < # ---------
11 < # new(ConfigArea,Builder)
6 < # ParseBuildFile($base,$path,$file)
7 < # ParseBuildFileExport(filename)
8 < # BlockParse(Block) : perform a parse to modify the block
9 < # BlockClassPath() : Return the class path
10 < # ignore()      : return 1 if directory should be ignored 0 otherwise
11 < # classname()   : get/set the associated class
12 < # buildfile()   : get/set BuildFile location
13 < # makefile()    : get the generated makefile
14 <
9 > # Copyright: 2003 (C) Shaun Ashby
10 > #
11 > #--------------------------------------------------------------------
12   package BuildSystem::BuildFile;
16 use ActiveDoc::SimpleDoc;
17 use BuildSystem::ToolBox;
13   require 5.004;
14 + use Exporter;
15 + use ActiveDoc::SimpleDoc;
16  
17 < BEGIN {
18 < $buildfile="BuildFile";
22 < }
23 <
24 < sub new {
25 <        my $class=shift;
26 <        my $self={};
27 <        bless $self, $class;
28 <        $self->{area}=shift;
29 <        $self->{Builder}=shift;
30 <        $self->{toolbox}=$self->{area}->toolbox();
31 <        $self->{localtop}=$self->{area}->location();
32 <        # -- set RELEASTOP
33 <        my $rarea=$self->{area}->linkarea();
34 <        if ( ! defined $rarea ) {
35 <          $self->{releasetop}=$self->{localtop};
36 <        }
37 <        else {
38 <          $self->{releasetop}=$rarea->location();
39 <        }
40 <        $self->{releasetop}=$self->{area}->location();
41 <        $self->{Arch}=1;
42 <        push @{$self->{ARCHBLOCK}}, $self->{Arch};
43 <        return $self;
44 < }
45 <
46 < sub buildfile {
47 <        my $self=shift;
48 <        if ( @_ ) {
49 <          $self->{buildfile}=shift;
50 <        }
51 <        return $self->{buildfile};
52 < }
53 <
54 < sub makefile {
55 <        my $self=shift;
56 <        if ( @_ ) {
57 <          $self->{makefile}=shift;
58 <        }
59 <        return $self->{makefile};
60 < }
61 <
62 < sub BlockParse {
63 <        my $self=shift;
64 <        $self->{block}=shift;
65 <
66 <        # -- set up the block parse
67 <        my $switch=$self->_initswitcher();
68 <        my $parse="block";
69 <        $switch->newparse($parse);
70 <        $switch->addignoretags($parse);
71 <        $switch->addtag($parse,"BuildParam", \&BuildBlock_start, $self);
72 <        $switch->filetoparse($self->buildfile());
73 <
74 <        # -- parse away
75 <        $self->{switch}=$switch;
76 <        $switch->parse("block");
77 < }
78 <
79 < sub Parsetofh {
80 <        my $self=shift;
81 <        my $fh=shift;
82 <        $self->{buildblock}=shift;
83 <
84 <        # -- set up for parse
85 <        @{$self->{filehandlestack}}=($fh);
86 <        $self->{switch}->filetoparse($self->buildfile());
87 <        *GNUmakefile=$fh;
88 <
89 <        # -- generate makefile
90 <        $self->{switch}->parse("makebuild"); # sort out supported tags
91 <
92 <        # -- Clean up
93 <        close GNUmakefile;
94 < }
95 <
96 < sub ignore {
97 <        my $self=shift;
98 <        return (defined $self->{ignore})?$self->{ignore}:0;
99 < }
100 <
101 < sub _initswitcher {
102 <        my $self=shift;
103 <        my $switch=ActiveDoc::SimpleDoc->new();
104 <        my $parse="makebuild";
105 <        $switch->newparse($parse);
106 <        $switch->addignoretags($parse);
107 <        $self->_commontags($switch,$parse);
108 <        #$switch->addtag($parse,"Build", \&Build_start, $self);
109 <        $switch->addtag($parse,"none",
110 <                                        \&OutToMakefile,$self,
111 <                                        \&OutToMakefile, $self,
112 <                                        "", $self);
113 <        $switch->addtag($parse,"Bin",
114 <                                        \&Bin_start,$self,
115 <                                        \&OutToScreen, $self,
116 <                                        "", $self);
117 <         $switch->addtag($parse,"ProductStore",
118 <                                        \&Store_start,$self,
119 <                                        "", $self,
120 <                                        "", $self);
121 <        $switch->addtag($parse,"LibType",
122 <                                        \&LibType_Start,$self,
123 <                                        \&LibType_text, $self,
124 <                                        \&LibType_end,$self);
125 <        $switch->addtag($parse,"ConfigurationClass",
126 <                                        \&Class_StartTag,$self,
127 <                                        \&OutToMakefile, $self,
128 <                                        "", $self);
129 <        $switch->addtag($parse,"ClassPath",
130 <                                        \&setBlockClassPath,$self,
131 <                                        \&OutToMakefile, $self,
132 <                                        "", $self);
133 <        $switch->addtag($parse,"AssociateGroup",
134 <                                        "",$self,
135 <                                        \&AssociateGroup,$self,
136 <                                        "", $self);
137 <        $switch->addtag($parse,"Environment",
138 <                                        \&Environment_start,$self,
139 <                                        \&OutToMakefile, $self,
140 <                                        \&Environment_end,$self);
141 <        $switch->addtag($parse,"Export",
142 <                                        \&export_start,$self,
143 <                                        \&OutToMakefile, $self,
144 <                                        \&export_end,$self);
145 <        return $switch;
146 < }
147 <
148 < sub _commontags {
149 <        my $self=shift;
150 <        my $switch=shift;
151 <        my $parse=shift;
152 <
153 <        $switch->grouptag("Export",$parse);
154 <        $switch->addtag($parse,"Use",\&Use_start,$self,
155 <                                               \&OutToMakefile, $self,
156 <                                                "", $self);
157 <        $switch->addtag($parse,"Group",\&Group_start,$self,
158 <                                               \&OutToMakefile, $self,
159 <                                                "", $self);
160 <        $switch->grouptag("Group",$parse);
161 <        $switch->addtag($parse,"External",
162 <                                        \&External_StartTag,$self,
163 <                                        \&OutToMakefile, $self,
164 <                                        "", $self);
165 <        $switch->addtag($parse,"lib",
166 <                                        \&lib_start,$self,
167 <                                        \&OutToMakefile, $self,
168 <                                        "", $self);
169 <        $switch->addtag($parse,"Architecture",
170 <                                        \&Arch_Start,$self,
171 <                                        \&OutToMakefile, $self,
172 <                                        \&Arch_End,$self);
173 <        $switch->addtag($parse,"INCLUDE_PATH",
174 <                                        \&IncludePath_Start,$self,
175 <                                        \&OutToMakefile, $self,
176 <                                        "",$self);
177 <        return $switch;
178 < }
179 <
180 < sub GenerateMakefile {
181 <        my $self=shift;
182 <        my $infile=shift;
183 <        my $outfile=shift;
184 <
185 <        $self->{switch}=$self->_initswitcher();
186 <        $self->{switch}->filetoparse($infile);
187 <
188 <        # open a temporary gnumakefile to store output.
189 <        my $fh=FileHandle->new();
190 <        open ( $fh, ">$outfile") or die "Unable to open $outfile for output ".
191 <                                                                "$!\n";
192 <        @{$self->{filehandlestack}}=($fh);
193 <
194 <        #  -- make an alias
195 <        *GNUmakefile=$fh;
196 <        if ( -e $ENV{LatestBuildFile} ) {
197 <          print GNUmakefile "include $ENV{LatestBuildFile}\n";
198 <        }
199 <        $self->{switch}->parse("makebuild"); # sort out supported tags
200 <        close GNUmakefile;
201 <        return $outfile;
202 < }
203 <
204 < sub ParseBuildFile {
205 <        my $self=shift;
206 <        my $base=shift;
207 <        my $path=shift;
208 <        my $filename=shift @_;
209 <        my $fullfilename;
210 <        if ( $filename!~/^\// ) {
211 <         $fullfilename="$base/$path/$filename";
212 <        }
213 <        else {
214 <         $fullfilename=$filename;
215 <        }
216 <        $self->{path}=$path;
217 <        #print "Processing $fullfilename\n";
218 <        $numbins=0;
219 <        $self->{envnum}=0;
220 <        $self->{envlevel}=0;
221 <        $self->{makefile}="$self->{localtop}/$ENV{INTwork}/$self->{path}/".
222 <                                                                "BuildFile.mk";
223 <        $self->{currentenv}=$self->{makefile};
224 <        $self->{switch}=$self->_initswitcher();
225 <        $self->{switch}->filetoparse($fullfilename);
226 <
227 < #       $self->{switch}->{Strict_no_cr}='no';
228 <        #open a temporary gnumakefile to store output.
229 <        use Utilities::AddDir;
230 <        AddDir::adddir("$self->{localtop}/$ENV{INTwork}/$self->{path}");
231 <        $ENV{LatestBuildFile}=$self->GenerateMakefile($fullfilename,
232 <        $self->{localtop}."/".$ENV{INTwork}."/".$self->{path}."/BuildFile.mk");
233 < }
234 <
235 < sub classname {
236 <        my $self=shift;
237 <        if ( @_ ) {
238 <          $self->{classname}=shift;
239 <        }
240 <        return $self->{classname};
241 < }
242 <
243 < sub ParseBuildFile_Export {
244 <        my $self=shift;
245 <        my $filename=shift;
246 <        my $bf=BuildSystem::BuildFile->new($self->{area},$self->{Builder});
247 <        if ( defined $self->{remoteproject} ) {
248 <           $bf->{remoteproject}=$self->{remoteproject};
249 <        }
250 <        $bf->_parseexport($filename);
251 <        undef $bf;
252 < }
253 <
254 < sub _location {
255 <        my $self=shift;
256 <        use File::Basename;
257 <
258 <        return dirname($self->{switch}->filetoparse());
259 < }
260 <
261 < sub _parseexport {
262 <        my $self=shift;
263 <        my $filename=shift;
264 <
265 <        my $switchex=ActiveDoc::SimpleDoc->new();
266 <        $switchex->filetoparse($filename);
267 <        $switchex->newparse("export");
268 <        $switchex->addignoretags("export");
269 <        $switchex->addtag("export","Export",
270 <                                        \&export_start_export,$self,
271 <                                        \&OutToMakefile, $self,
272 <                                        \&export_end_export,$self);
273 <        $self->_commontags($switchex,"export");
274 <        $switchex->allowgroup("__export","export");
275 < #       $switchex->{Strict_no_cr}='no';
276 <        $self->{switch}=$switchex;
277 <        $switchex->parse("export"); # sort out supported tags
278 < }
279 <
280 < sub _pushremoteproject {
281 <        my $self=shift;
282 <        my $path=shift;
283 <        
284 <        if ( defined $self->{remoteproject} ) {
285 <          push @{$self->{rpstack}}, $self->{remoteproject};
286 <        }
287 <        $self->{remoteproject}=$path;
288 < }
289 <
290 < sub _popremoteproject {
291 <        my $self=shift;
292 <        if ( $#{$self->{rpstack}} >=0 ) {
293 <          $self->{remoteproject}=pop @{$self->{rpstack}};
294 <        }
295 <        else {
296 <          undef $self->{remoteproject};
297 <        }
298 < }
299 <
300 < sub _toolmapper {
301 <        my $self=shift;
302 <        if ( ! defined $self->{mapper} ) {
303 <           require BuildSystem::ToolMapper;
304 <           $self->{mapper}=BuildSystem::ToolMapper->new();
305 <        }
306 <        return $self->{mapper};
307 < }
308 <
309 <
310 < # ---- Tag routines
311 <
312 < #-- Override a class type with the <ConfigurationClass type=xxx> tag
313 < #   the type tag will pick up a pre-defined class type from project space.
314 <
315 < sub Class_StartTag {
316 <        my $self=shift;
317 <        my $name=shift;
318 <        my $hashref=shift;
319 <        
320 <        if ( $self->{Arch} ) {
321 <         if ( defined $$hashref{'type'} ) {
322 <                 $self->classname($$hashref{'type'});
323 <         }
324 <        }
325 < }
326 <
327 < sub IncludePath_Start {
328 <        my $self=shift;
329 <        my $name=shift;
330 <        my $hashref=shift;
331 <
332 <        $self->{switch}->checktag( $name, $hashref, 'path');
333 <        if ( $self->{Arch} ) {
334 <          print GNUmakefile "INCLUDE+=".$self->_location()."/".
335 <                                                $$hashref{'path'}."\n";
336 <        }
337 < }
338 <
339 < #
340 < # --- <Build class=> tag
17 > @ISA=qw(Exporter);
18 > @EXPORT_OK=qw( );
19   #
20 + sub new()
21 +   ###############################################################
22 +   # new                                                         #
23 +   ###############################################################
24 +   # modified : Wed Dec  3 19:03:22 2003 / SFA                   #
25 +   # params   :                                                  #
26 +   #          :                                                  #
27 +   # function :                                                  #
28 +   #          :                                                  #
29 +   ###############################################################
30 +   {
31 +   my $proto=shift;
32 +   my $class=ref($proto) || $proto;
33 +   $self={};
34 +   bless $self,$class;
35 +   $self->{DEPENDENCIES} = {};
36 +   $self->{content} = {};
37 +   $self->{scramdoc}=ActiveDoc::SimpleDoc->new();
38 +   $self->{scramdoc}->newparse("builder",__PACKAGE__,'Subs');
39 +   return $self;
40 +   }
41 +
42 + sub parse()
43 +   {
44 +   my $self=shift;
45 +   my ($filename)=@_;
46 +   $self->{scramdoc}->filetoparse($filename);
47 +   $self->{scramdoc}->parse("builder");
48 +   # We're done with the SimpleDoc object so delete it:
49 +   delete $self->{scramdoc};
50 +   }
51 +
52 + sub classpath()
53 +   {
54 +   my ($object,$name,%attributes)=@_;
55 +   # The getter part:
56 +   if (ref($object) eq __PACKAGE__)
57 +      {
58 +      return $self->{content}->{CLASSPATH};
59 +      }
60 +  
61 +   $self->{nested} == 1 ? push(@{$self->{tagcontent}->{CLASSPATH}}, $attributes{'path'})
62 +      : push(@{$self->{content}->{CLASSPATH}}, $attributes{'path'});
63 +   }
64 +
65 + sub productstore()
66 +   {
67 +   my ($object,$name,%attributes)=@_;
68 +   # The getter part:
69 +   if (ref($object) eq __PACKAGE__)
70 +      {
71 +      # Return an array of ProductStore hashes:
72 +      return $self->{content}->{PRODUCTSTORE};
73 +      }
74 +  
75 +   $self->{nested} == 1 ? push(@{$self->{tagcontent}->{PRODUCTSTORE}}, \%attributes)
76 +      : push(@{$self->{content}->{PRODUCTSTORE}}, \%attributes) ;
77 +   }
78 +
79 + sub include()
80 +   {
81 +   my $self=shift;
82 +   # Return an array of required includes:
83 +   return $self->{content}->{INCLUDE};
84 +   }
85 +
86 + sub include_path()
87 +   {
88 +   my ($object,$name,%attributes)=@_;
89 +   $self->{nested} == 1 ? push(@{$self->{tagcontent}->{INCLUDE}}, $attributes{'path'})
90 +      : push(@{$self->{content}->{INCLUDE}}, $attributes{'path'});
91 +   }
92 +
93 + sub use()
94 +   {
95 +   my $object=shift;
96 +   # The getter part:
97 +   if (ref($object) eq __PACKAGE__)
98 +      {
99 +      # Add or return uses (package deps):
100 +      @_ ? push(@{$self->{content}->{USE}},@_)
101 +         : @{$self->{content}->{USE}};
102 +      }
103 +   else
104 +      {
105 +      my ($name,%attributes)=@_;
106 +      $self->{DEPENDENCIES}->{$attributes{'name'}} = 1;
107 +      $self->{nested} == 1 ? push(@{$self->{tagcontent}->{USE}}, $attributes{'name'})
108 +         : push(@{$self->{content}->{USE}}, $attributes{'name'});
109 +      }
110 +   }
111 +
112 + sub architecture()
113 +   {
114 +   my ($object,$name,%attributes)=@_;
115 +   $self->pushlevel(\%attributes); # Set nested to 1;
116 +   }
117 +
118 + sub architecture_()
119 +   {
120 +   $self->{content}->{ARCH}->{$self->{id}->{'name'}}=$self->{tagcontent};
121 +   $self->poplevel();
122 +   }
123 +
124 + sub export()
125 +   {
126 +   $self->pushlevel(); # Set nested to 1;
127 +   }
128 +
129 + sub export_()
130 +   {
131 +   $self->{content}->{EXPORT} = $self->{tagcontent};
132 +   $self->poplevel();
133 +   }
134 +
135 + sub lib()
136 +   {
137 +   my ($object,$name,%attributes)=@_;
138 +   # The getter part:
139 +   if (ref($object) eq __PACKAGE__)
140 +      {
141 +      # Return an array of required libs:
142 +      return $self->{content}->{LIB};      
143 +      }
144 +  
145 +   my $libname;
146 +  
147 +   if (exists($attributes{'position'}))
148 +      {
149 +      if ($attributes{'position'} eq 'first')
150 +         {
151 +         $libname = "F:".$attributes{'name'};
152 +         }
153 +      else
154 +         {
155 +         # There was a position entry but it didn't make sense:
156 +         $libname = $attributes{'name'};
157 +         }
158 +      }
159 +   else
160 +      {
161 +      $libname = $attributes{'name'};
162 +      }
163 +   # We have a libname, add it to the list:
164 +   $self->{nested} == 1 ? push(@{$self->{tagcontent}->{LIB}}, $libname)
165 +      : push(@{$self->{content}->{LIB}}, $libname);
166 +   }
167 +
168 + sub libtype()
169 +   {
170 +   my ($object,$name,%attributes)=@_;
171 +   # The getter part:
172 +   if (ref($object) eq __PACKAGE__)
173 +      {
174 +      # Return an array of required libs:
175 +      return $self->{content}->{LIBTYPE};      
176 +      }
177 +
178 +   $self->{nested} == 1 ? push(@{$self->{tagcontent}->{LIBTYPE}}, $attributes{'type'})
179 +      : push(@{$self->{content}->{LIBTYPE}}, $attributes{'type'});
180 +   }
181 +
182 + sub skip()
183 +   {
184 +   my ($object,$name,%attributes)=@_;
185 +   $self->{nested} == 1 ? $self->{tagcontent}->{SKIPPEDDIRS} = [ 1 ]
186 +      : $self->{content}->{SKIPPEDDIRS} = [ 1 ];
187 +   }
188 +
189 + sub skip_message()
190 +   {
191 +   my ($object,$name,@message) = @_;
192 +   # Save any message text between <skip> tags:
193 +   if ($#message > -1)
194 +      {
195 +      $self->{nested} == 1 ? push(@{$self->{tagcontent}->{SKIPPEDDIRS}}, [ @message ])
196 +         : push(@{$self->{content}->{SKIPPEDDIRS}}, [ @message ]);
197 +      }
198 +   }
199 +
200 + sub skip_()
201 +   {
202 +   my ($object,$name)=@_;
203 +   }
204 +
205 + sub makefile()
206 +   {
207 +   my ($object,$name,%attributes)=@_;
208 +   # The getter part:
209 +   if (ref($object) eq __PACKAGE__)
210 +      {
211 +      # Return Makefile content:
212 +      return $self->{content}->{MAKEFILE};
213 +      }
214 +  
215 +   # Set our own Char handler so we can collect the content
216 +   # of the Makefile tag:
217 +   $object->setHandlers(Char => \&makefile_content);
218 +   $self->{makefilecontent} = [];
219 +   }
220 +
221 + sub makefile_content()
222 +   {
223 +   my ($object, @strings) = @_;
224 +   push(@{$self->{makefilecontent}},@strings);
225 +   }
226 +
227 + sub makefile_()
228 +   {
229 +   my ($object,$name)=@_;
230 +   $self->{nested} == 1 ? push(@{$self->{tagcontent}->{MAKEFILE}}, join('',@{$self->{makefilecontent}}))
231 +      : push(@{$self->{content}->{MAKEFILE}}, join('',@{$self->{makefilecontent}}));
232 +   delete $self->{makefilecontent};
233 +   # Unset the Char handler to revert to the default behaviour:
234 +   $object->setHandlers(Char => 0);
235 +   }
236 +
237 + sub define_group()
238 +   {
239 +   my ($object,$name,%attributes)=@_;
240 +   $self->pushlevel(\%attributes); # Set nested to 1;
241 +   }
242 +
243 + sub define_group_()
244 +   {
245 +   $self->{content}->{DEFINED_GROUP}->{$self->{id}->{'name'}}=$self->{tagcontent};
246 +   $self->poplevel();
247 +   }
248 +
249 + sub group()
250 +   {
251 +   my $object=shift;
252 +   # The getter part:
253 +   if (ref($object) eq __PACKAGE__)
254 +      {
255 +      # Add or return groups:
256 +      @_ ? push(@{$self->{content}->{GROUP}},@_)
257 +         : @{$self->{content}->{GROUP}};
258 +      }
259 +   else
260 +      {
261 +      my ($name,%attributes)=@_;
262 +      $self->{nested} == 1 ? push(@{$self->{tagcontent}->{GROUP}}, $attributes{'name'})
263 +         : push(@{$self->{content}->{GROUP}}, $attributes{'name'});
264 +      }
265 +   }
266 +
267 + sub flags()
268 +   {
269 +   my ($object,$name,%attributes)=@_;
270 +   # The getter part:
271 +   if (ref($object) eq __PACKAGE__)
272 +      {
273 +      # Return an array of ProductStore hashes:
274 +      return $self->{content}->{FLAGS};
275 +      }
276 +  
277 +   # Extract the flag name and its value:
278 +   my ($flagname,$flagvaluestring) = each %attributes;
279 +   $flagname =~ tr/[a-z]/[A-Z]/; # Keep flag name uppercase
280 +   chomp($flagvaluestring);
281 +   # Split the value on whitespace so we can push all
282 +   # individual flags into an array:
283 +   my @flagvalues = split(' ',$flagvaluestring);
284 +   # Is current tag within another tag block?
285 +   if ($self->{nested} == 1)
286 +      {
287 +      # Check to see if the current flag name is already stored in the hash. If so,
288 +      # just add the new values to the array of flag values:
289 +      if (exists ($self->{tagcontent}->{FLAGS}->{$flagname}))
290 +         {
291 +         push(@{$self->{tagcontent}->{FLAGS}->{$flagname}},@flagvalues);
292 +         }
293 +      else
294 +         {
295 +         $self->{tagcontent}->{FLAGS}->{$flagname} = [ @flagvalues ];
296 +         }
297 +      }
298 +   else
299 +      {
300 +      if (exists ($self->{content}->{FLAGS}->{$flagname}))
301 +         {
302 +         push(@{$self->{content}->{FLAGS}->{$flagname}},@flagvalues);
303 +         }
304 +      else
305 +         {
306 +         $self->{content}->{FLAGS}->{$flagname} = [ @flagvalues ];
307 +         }
308 +      }
309 +   }
310  
311 < #
312 < # Parameter collection
313 < #
314 < sub BuildBlock_start {
315 <        my $self=shift;
316 <        my $name=shift;
317 <        my $hashref=shift;
318 <
319 <
320 <        my $blockobjid=$self->__blockobjid($hashref);
321 <
322 <        if ( $self->{Arch} ) {
323 <
324 <           # -- get any objects that match
325 <           my $inheritobj=$self->{block}->getobj($blockobjid);
326 <
327 <           # -- create an object with inherited properties
328 <           my $obj;
329 <           if ( ! defined $inheritobj ) {
362 <               # -- check we have a lookup for the class type
363 <               my $mapper=$self->_toolmapper();
364 <               if ( ! $mapper->exists($$hashref{'class'}) ) {
365 <                 $self->{switch}->parseerror("Unknown class : ".
366 <                                                        $$hashref{'class'});
367 <               }
368 <               $obj=BuildSystem::BuildClass->new();
369 <           }
370 <           else {
371 <               # -- inherit the properties from class with the same id class
372 <               $obj=$inheritobj->child();
373 <           }
374 <
375 <           # -- add changes from our tag
376 <           $obj->paramupdate($hashref);
377 <
378 <           # -- store the new object in the block
379 <           $self->{block}->setobj($obj,$blockobjid);
380 <        }
381 < }
382 <
383 < sub BuilderClass_buildmakefile {
384 <        my $self=shift;
385 <        my $name=shift;
386 <        my $hashref=shift;
387 <
388 <        my $blockobjid=$self->__blockobjid($hashref);
389 <
390 <        if ( $self->{Arch} ) {
391 <           # -- get the matching block object
392 <           my $blockobj=$self->{buildblock}->getobj($blockobjid);
393 <
394 <           # -- top level buildfile
395 <           my $fh=$self->{filehandlestack}[0];
396 <
397 <           # -- var initialisation
398 <           my @deftypes=();
399 <           my $buildname="";
400 <           my @types=$self->_toolmapper()->types($$hashref{'class'});
401 <
402 <           # -- error checking
403 <           if ( ! defined $blockobj->param("default") ) {
404 <             $self->error("No default build parameter defined for ".
405 <                $$hashref{'class'}." ".$$hashref{'id'});
406 <           }
407 <           if ( ! defined $blockobj->param("name") ) {
408 <             $self->error("\"name\" parameter defined for ".
409 <                $$hashref{'class'}." ".$$hashref{'id'});
410 <           }
411 <
412 <
413 <           foreach $param ( $blockobj->paramlist() ) {
414 <             # -- check for params that need special handling
415 <             if ( $param eq "default" ) {
416 <                @deftypes=split /,/, $param;
417 <             }
418 <             elsif ( $param eq "name" ) {
419 <                $buildname=$blockobj->param($param);
420 <             }
421 <             else {
422 <                # -- simple transfer of block object parameters to makefile
423 <                print $fh $param.":=".$blockobj->param($param)."\n";
424 <             }
425 <           }
426 <
427 <           # -- construct the targets in the top makefile
428 <           $self->_generatedefaulttargets($fh,$$hashref{'class'},@deftypes);
429 <           $self->_generatetypetargets($fh,$$hashref{'class'},$buildname,@types);
430 <        }
431 < }
432 <
433 < sub _blockobjid {
434 <        my $self=shift;
435 <        my $hashref=shift;
436 <
437 <        $self->{switch}->checktag($name,$hashref,'class');
438 <        $self->{switch}->checktag($name,$hashref,'id');
439 <        my $blockobjid="bc_".$$hashref{'class'},"_".$$hashref{'id'};
440 <
441 <        return $blockobjid;
442 < }
443 <
444 < sub Build_start {
445 <        my $self=shift;
446 <        my $name=shift;
447 <        my $hashref=shift;
448 <
449 <        $self->{switch}->checktag($name,$hashref,'class');
450 <        $self->{switch}->checktag($name,$hashref,'id');
451 <        if ( $self->{Arch} ) {
452 <
453 <          # -- determine the build products name
454 <          my $name;
455 <          if ( exists $$hashref{'name'} ) {
456 <            $name=$$hashref{'name'};
457 <          }
458 <          else {
459 <            $self->{switch}->parseerror("No name specified for build product");
460 <            #$name="\$(buildname)";
461 <          }
462 <
463 <          # -- check we have a lookup for the class type
464 <          my $mapper=$self->_toolmapper();
465 <          if ( ! $mapper->exists($$hashref{'class'}) ) {
466 <            $self->{switch}->parseerror("Unknown class : ".$$hashref{'class'});
467 <          }
468 <          else {
469 <           my @types=$self->_toolmapper()->types($$hashref{'class'});
470 <           my @deftypes=$self->_toolmapper()->defaulttypes($$hashref{'class'});
471 <
472 <           my $fh=$self->{filehandlestack}[0];
473 <           my @targets=();
474 <
475 <           # -- generate generic targets
476 <           print $fh "ifndef _BuildLink_\n";
477 <           print $fh "# -- Generic targets\n";
478 <           push @targets, $$hashref{'class'};
479 <           foreach $dtype ( @deftypes ) {
480 <            print $fh $$hashref{'class'}."::".$$hashref{'class'}."_".
481 <                                                                $dtype."\n";
482 <           }
483 <           print $fh "\n";
484 <
485 <           # -- generate targets for each type
486 <           foreach $type ( @types ) {
487 <
488 <            # -- generic name for each type
489 <            my $pattern=$$hashref{'class'}."_".$type;
490 <            my $dirname=$$hashref{'class'}."_".$type."_".$name;
491 <            print $fh "# ------ $pattern rules ---------------\n";
492 <            print $fh $$hashref{'class'}."_".$type."::".$$hashref{'class'}.
493 <                                                        "_".$type."_$name\n\n";
494 <
495 <            # -- create a new directory for each type
496 <            push @targets, $pattern;
497 <            my $dirname=$$hashref{'class'}."_".$type."_".$name;
498 <            my $here="$ENV{LOCALTOP}/$ENV{INTwork}/".$self->{path}."/".$dirname;
499 <            my $makefile=$here."/BuildFile.mk";
500 < #           AddDir::adddir($here);
501 <
502 <            # -- create link targets to the directory
503 <            push @targets, $dirname;
504 <            print $fh "# -- Link Targets to $type directories\n";
505 <            print $fh "$dirname: make_$dirname\n";
506 <            print $fh "\t\@cd $here; \\\n";
507 <            print $fh "\t\$(MAKE) LatestBuildFile=$makefile _BuildLink_=1".
508 <                        " workdir=$here ".
509 <                        " -f \$(TOOL_HOME)/basics.mk datestamp \$\@; \n\n";
510 <
511 <            # -- write target to make makefile for each directory
512 <            print $fh "# -- Build target directories\n";
513 <            print $fh "make_$dirname:\n";
514 <            print $fh "\tif [ ! -e \"$makefile\" ]; then \\\n";
515 <            print $fh "\t if [ ! -d \"$here\" ]; then \\\n";
516 <            print $fh "\t  mkdir $here; \\\n";
517 <            print $fh "\t fi;\\\n";
518 <            print $fh "\t cd $dirname; \\\n";
519 <            print $fh "\t echo include ".$self->{currentenv}." > ".
520 <                                                        "$makefile; \\\n";
521 <            print $fh "\t echo VPATH+=$ENV{LOCALTOP}/".$self->{path}.
522 <                                        " >> $makefile; \\\n";
523 <            print $fh "\t echo buildname=$name >> $makefile;\\\n";
524 <            print $fh "\t echo ".$dirname.":".$pattern." >> $makefile;\\\n";
525 <            if ( defined (my @file=$mapper->rulesfile($$hashref{'class'})) ) {
526 <             foreach $f ( @file ) {
527 <              print $fh "\t echo -include $f >> $makefile; \\\n";
528 <             }
311 > sub allflags()
312 >   {
313 >   my $self=shift;
314 >   # Return hash data for flags:
315 >   return $self->{content}->{FLAGS};
316 >   }
317 >
318 > sub archspecific()
319 >   {
320 >   my $self=shift;
321 >  
322 >   # Check to see if there is arch-dependent data. If so, return it:
323 >   if ((my $nkeys=keys %{$self->{content}->{ARCH}}) > 0)
324 >      {
325 >      while (my ($k,$v) = each %{$self->{content}->{ARCH}})
326 >         {
327 >         if ( $ENV{SCRAM_ARCH} =~ /$k.*/ )
328 >            {
329 >            return $self->{content}->{ARCH}->{$k};
330              }
530            print $fh "\tfi\n";
531            print $fh "\n";
532 #           print $typefile "$name :\n";
533 #           print $typefile "\t\$(_quietbuild_)";
534 #           print $typefile $mapper->template($$hashref{'class'},$type)."\n";
535 #           print $typefile "\t\$(_quietstamp_)";
536 #           print $typefile "\$(SCRAMPERL) \$(SCRAM_HOME)/src/scramdatestamp \$@.ds \$@ \$^\n";
537
538            # -- cleaning targets
539            push @targets, "clean_$dirname";
540            print $fh "# -- cleaning targets\n";
541            print $fh "clean::clean_$dirname\n";
542            print $fh "clean_".$dirname."::\n";
543            print $fh "\t\@echo cleaning $dirname\n";
544            print $fh "\t\@if [ -d $here ]; then \\\n";
545            print $fh "\tcd $here; \\\n";
546            print $fh "\t\$(MAKE) LatestBuildFile=$makefile workdir=".
547                        $here." _BuildLink_=1 -f ".
548                        "\$(TOOL_HOME)/basics.mk clean; \\\n";
549            print $fh "\tfi\n\n";
550
551
552          }
553          # -- help targets
554          print $fh "helpheader::\n";
555          print $fh "\t\@echo Targets available:\n";
556          print $fh "\t\@echo ------------------\n\n";
557          print $fh "help::helpheader\n";
558          foreach $target ( @targets ) {
559            print $fh "help::\n";
560            print $fh "\t\@echo $target\n"
561          }
562          print $fh "endif\n";
563         } # end else
564        }
565 }
566
567 sub Bin_start {
568        my $self=shift;
569        my $name=shift;
570        my $hashref=shift;
571
572        my $fileclass;
573        my @tools;
574        my $tool;
575        my $filename;
576        my $objectname;
577        
578        $self->{switch}->checktag($name,$hashref,'file');
579        if ( $self->{Arch} ) {
580        if ( ! defined $$hashref{name} ) {
581                ($$hashref{name}=$$hashref{file})=~s/\..*//;
582        }
583        ($filename=$$hashref{file})=~s/\..*//;
584
585        # Create a new directory for each binary target
586        my $dirname="bin_".$$hashref{name};
587        AddDir::adddir("$ENV{LOCALTOP}/$ENV{INTwork}/".$self->{path}."/$dirname");
588        open (binGNUmakefile,
589           ">$ENV{LOCALTOP}/$ENV{INTwork}/".$self->{path}."/$dirname/BuildFile.mk") or die           "Unable to make $ENV{LOCALTOP}/$ENV{INTwork}/$self->{path}/$dirname/".
590           "BuildFile.mk $!\n";
591
592        # Create the link targets
593        $numbins++;
594        my $fh=$self->{filehandlestack}[0];
595        print $fh <<ENDTEXT;
596
597 # Link Targets to binary directories
598 ifdef BINMODE
599 # We dont want to build a library here
600 override files:=
601 endif
602 ifndef BINMODE
603
604 define stepdown_$$hashref{'name'}
605 if [ -d "$ENV{LOCALTOP}/$ENV{INTwork}/$self->{path}/$dirname" ]; then \\
606 cd $ENV{LOCALTOP}/$ENV{INTwork}/$self->{path}/$dirname; \\
607 \$(MAKE) BINMODE=true LatestBuildFile=$ENV{LOCALTOP}/$ENV{INTwork}/$self->{path}/$dirname/BuildFile.mk workdir=\$(workdir)/$dirname -f \$(TOOL_HOME)/basics.mk datestamp \$\@; \\
608 fi
609 endef
610
611 define stepdown2_$$hashref{'name'}
612 if [ -d "$ENV{LOCALTOP}/$ENV{INTwork}/$self->{path}/$dirname" ]; then \\
613 cd $ENV{LOCALTOP}/$ENV{INTwork}/$self->{path}/$dirname; \\
614 \$(MAKE) BINMODE=true LatestBuildFile=$ENV{LOCALTOP}/$ENV{INTwork}/$self{path}/$dirname/BuildFile.mk workdir=\$(workdir)/$dirname -f \$(TOOL_HOME)/basics.mk datestamp \$\*; \\
615 fi
616
617 endef
618
619 bin_$$hashref{'name'}_%:: dummy
620        \@\$(stepdown2_$$hashref{'name'})
621
622 $$hashref{'name'}_%:: dummy
623        \@\$(stepdown_$$hashref{'name'})
624
625 help bin bin_debug bin_debug_local bin_insure bin_Insure clean $$hashref{'name'}:: dummy
626        \@\$(stepdown_$$hashref{'name'})
627
628 binfiles+=$$hashref{'file'}
629 locbinfiles+=$dirname/$$hashref{'file'}
630 endif
631
632
633 ENDTEXT
634
635
636 # the binary specifics makefile
637        print binGNUmakefile "include ".$self->{currentenv}."\n";
638        print binGNUmakefile "VPATH+=$ENV{LOCALTOP}/$self{path}\n";
639
640 # alias for bin_Insure
641        print binGNUmakefile <<ENDTEXT;
642
643 bin_insure:bin_Insure
644 ifdef MAKETARGET_bin_insure
645 MAKETARGET_$$hashref{name}_Insure=1
646 endif
647
648 # debuggging target
649 $$hashref{'name'}_echo_% :: echo_%
650
651 # help targets
652 help::
653 \t\@echo Targets For $$hashref{'name'}
654 \t\@echo -------------------------------------
655 \t\@echo $$hashref{'name'}  - default build
656 \t\@echo bin_$$hashref{'name'}_clean - executable specific cleaning
657 ENDTEXT
658
659 # Make generic rules for each type
660        $targettypes={
661                "bin" => 'o',
662                "bin_debug" => 'd',
663                "bin_debug_local" => 'l_d',
664                "bin_Insure" => 'Insure'
665        };
666        #
667        foreach $target ( keys %$targettypes ) {
668          print binGNUmakefile <<ENDTEXT;
669
670 # Type $target specifics
671 ifdef MAKETARGET_$target
672 MAKETARGET_$$hashref{name}_$$targettypes{$target}=1
673 endif
674 $target ::$$hashref{name}_$$targettypes{$target}
675
676 bintargets+=$$hashref{name}_$$targettypes{$target}
677 help::
678 \t\@echo $$hashref{name}_$$targettypes{$target}
679 clean::
680 \t\@if [ -f \$(binarystore)/$$hashref{name}_$$targettypes{$target} ]; then \\
681 \techo Removing \$(binarystore)/$$hashref{name}; \\
682 \trm \$(binarystore)/$$hashref{name}_$$targettypes{$target}; \\
683 \tfi
684
685 ENDTEXT
686          ($objectname=$$hashref{file})=~s/\..*/_$$targettypes{$target}\.o/;
687          ${"objectname_$$targettypes{$target}"}=$objectname;
688          print binGNUmakefile "$objectname:$$hashref{name}.dep\n";
689        } # end loop
690
691        print binGNUmakefile "$$hashref{name}_Insure.exe:.psrc\n";
692        print binGNUmakefile "$$hashref{name}_d.exe:$objectname_d\n";
693        print binGNUmakefile "\t\$(CClinkCmdDebug)\n";
694        print binGNUmakefile "\t\@\$(SCRAMPERL) \$(SCRAM_HOME)/src/scramdatestamp \$\@\.ds \$\@ \$\^\n";
695        print binGNUmakefile "$$hashref{name}_l_d.exe:$objectname_d\n";
696        print binGNUmakefile "\t\$(CClinkCmdDebugLocal)\n";
697        print binGNUmakefile "\t\@\$(SCRAMPERL) \$(SCRAM_HOME)/src/scramdatestamp \$\@\.ds \$\@ \$\^\n";
698        print binGNUmakefile "$$hashref{name}_Insure.exe:$objectname_Insure\n";
699        print binGNUmakefile "\t\$(CClinkCmdInsure)\n";
700        print binGNUmakefile "\t\@\$(SCRAMPERL) \$(SCRAM_HOME)/src/scramdatestamp \$\@\.ds \$\@ \$\^\n";
701        print binGNUmakefile "$$hashref{name}_o.exe:$objectname_o\n";
702        print binGNUmakefile "\t\$(CClinkCmd)\n";
703        print binGNUmakefile "\t\@\$(SCRAMPERL) \$(SCRAM_HOME)/src/scramdatestamp \$\@\.ds \$\@ \$\^\n";
704        print binGNUmakefile "$$hashref{name}.dep:$$hashref{file}\n";
705        print binGNUmakefile "-include $$hashref{name}.dep\n";
706 print binGNUmakefile <<ENDTEXT;
707 clean::
708 \t\@if [ -f \$(binarystore)/$$hashref{name} ]; then \\
709 \techo Removing \$(binarystore)/$$hashref{name}; \\
710 \trm \$(binarystore)/$$hashref{name}; \\
711 \tfi
712
713 $$hashref{name}_d.exe:\$(libslocal_d)
714 $$hashref{name}_o.exe:\$(libslocal)
715 ifdef MCCABE_DATA_DIR
716 $$hashref{name}_mccabe.exe: \$(libslocal_d) \$(MCCABE_DATA_DIR)/mccabeinstr/instplus.cpp
717 endif
718 $$hashref{name}_Insure.exe:\$(libslocal_I)
719 $$hashref{name}_d:$$hashref{name}_d.exe
720        \@cp $$hashref{name}_d.exe \$(binarystore)/$$hashref{name}
721 $$hashref{name}_l_d:$$hashref{name}_l_d.exe
722        \@cp $$hashref{name}_l_d.exe \$(binarystore)/$$hashref{name}
723 $$hashref{name}_Insure:$$hashref{name}_Insure.exe
724        \@cp $$hashref{name}_Insure.exe \$(binarystore)/$$hashref{name}_Insure
725 $$hashref{name}:$$hashref{name}_d.exe
726        \@mv $$hashref{name}_d.exe \$(binarystore)/$$hashref{name}
727 $$hashref{name}_o:$$hashref{name}_o.exe
728        \@mv $$hashref{name}_o.exe \$(binarystore)/$$hashref{name}
729 binfiles+=$$hashref{file}
730 ENDTEXT
731        }
732        close binGNUmakefile;
733 }
734
735 sub External_StartTag {
736        my $self=shift;
737        my $name=shift;
738        my $hashref=shift;
739        
740        my $tool;
741        if ( $self->{Arch} ) {
742        $self->{switch}->checktag($name,$hashref,'ref');
743
744        # -- oo toolbox stuff
745        # - get the appropriate tool object
746        $$hashref{'ref'}=~tr[A-Z][a-z];
747        if ( ! exists $$hashref{'version'} ) {
748         $tool=$self->{toolbox}->gettool($$hashref{'ref'});
749        }
750        else {
751         $tool=$self->{toolbox}->gettool($$hashref{'ref'},$$hashref{'version'});
752        }
753        if ( ! defined $tool ) {
754          $self->{switch}->parseerror("Unknown Tool Specified ("
755                                                        .$$hashref{'ref'}.")");
756        }
757
758        # -- old fashioned GNUmakefile stuff
759        print GNUmakefile $$hashref{'ref'};
760        if ( defined $$hashref{'version'} ) {
761                print GNUmakefile "_V_".$$hashref{'version'};
762        }
763        print GNUmakefile "=true\n";
764        
765        # -- Sub system also specified?
766        if ( exists $$hashref{'use'} ) {
767           # -- look for a buildfile
768           my @paths=$tool->getfeature("INCLUDE");
769           my $file="";
770           my ($path,$testfile);
771           foreach $path ( @paths ) {
772             $testfile=$path."/".$$hashref{'use'}."/BuildFile" ;
773             if ( -f $testfile ) {
774                $file=$testfile;
775                $self->_pushremoteproject($path);
776             }
777           }
778           if ( $file eq "" ) {
779             $self->{switch}->parseerror("Unable to find SubSystem $testfile");
780           }
781           $self->ParseBuildFile_Export($file);
782           $self->_popremoteproject();
331           }
332 <        }
333 < }      
334 <
335 < sub Group_start {
336 <        my $self=shift;
337 <        my $name=shift;
338 <        my $hashref=shift;
339 <        
340 <        $self->{switch}->checktag($name, $hashref, 'name');
341 <        if ( $self->{Arch} ) {
342 <        print GNUmakefile "GROUP_".$$hashref{'name'};
343 <        if ( defined $$hashref{'version'} ) {
344 <                print GNUmakefile "_V_".$$hashref{'version'};
345 <        }
346 <        print GNUmakefile "=true\n";
347 <        }
348 < }      
349 <
350 < sub Use_start {
351 <        my $self=shift;
352 <        my $name=shift;
353 <        my $hashref=shift;
354 <        my $filename;
355 <        use Utilities::SCRAMUtils;
356 <        
357 <        $self->{switch}->checktag($name, $hashref, "name");
358 <        if ( $self->{Arch} ) {
359 <        if ( exists $$hashref{'group'} ) {
360 <          print GNUmakefile "GROUP_".$$hashref{'group'}."=true\n";
361 <        }
362 <        if ( ! defined $self->{remoteproject} ) {
363 <          $filename=SCRAMUtils::checkfile(
364 <                "/$ENV{INTsrc}/$$hashref{name}/BuildFile");
365 <          # -- force a dependency check if local
366 <          if ( $filename=~/^$self->{localtop}/ ) {
367 <            $self->{Builder}->BuildDir($ENV{INTsrc}."/".$$hashref{'name'});
368 <          }
369 <        }
370 <        else {
371 <          $filename=$self->{remoteproject}."/$$hashref{name}/BuildFile";
372 <        print "trying $filename\n";
373 <          if ( ! -f $filename ) { $filename=""; };
374 <        }
375 <        if ( $filename ne "" ) {
376 <          $self->ParseBuildFile_Export( $filename );
377 <        }
378 <        else {
379 <           $self->{switch}->parseerror("Unable to detect Appropriate ".
380 <                "decription file for <$name name=".$$hashref{name}.">");
381 <        }
382 <        }
383 < }
384 <
385 < # List association groups between <AssociateGroup> tags
386 < # seperated by newlines or spaces
387 < sub AssociateGroup {
388 <        my $self=shift;
389 <        my $name=shift;
390 <        my $string=shift;
391 <        my $word;
392 <
393 <        if ( $self->{Arch} ) {
394 <        foreach $word ( (split /\s/, $string) ){
395 <                chomp $word;
396 <                next if /^#/;
397 <                if ( $word=~/none/ ) {
398 <                        $self->{ignore}=1;
399 <                }
400 <        }
401 <        }
402 < }
403 <
404 < sub Arch_Start {
405 <        my $self=shift;
406 <        my $name=shift;
407 <        my $hashref=shift;
408 <
409 <        $self->{switch}->checktag($name, $hashref,'name');
410 <        ( ($ENV{SCRAM_ARCH}=~/$$hashref{name}.*/) )? ($self->{Arch}=1)
411 <                                                : ($self->{Arch}=0);
412 <        push @{$self->{ARCHBLOCK}}, $self->{Arch};
413 < }
414 <
415 < sub Arch_End {
416 <        my $self=shift;
417 <        my $name=shift;
418 <
419 <        pop @{$self->{ARCHBLOCK}};
420 <        $self->{Arch}=$self->{ARCHBLOCK}[$#{$self->{ARCHBLOCK}}];
421 < }
422 <
423 < # Split up the Class Block String into a useable array
424 < sub _CutBlock {
425 <    my $self=shift;
426 <    my $string= shift @_;
427 <    @BlockClassA = split /\//, $string;
428 < }
429 <
430 < sub OutToMakefile {
431 <        my $self=shift;
432 <        my $name=shift;
433 <        my @vars=@_;
434 <
435 <        if ( $self->{Arch} ) {
436 <          print GNUmakefile @vars;
437 <        }
438 < }
439 <
440 < sub OutToScreen {
441 <        my $name=shift;
442 <        my @vars=@_;
443 <
444 <        if ( $self->{Arch} ) {
445 <          print @vars;
446 <        }
447 < }
448 < sub setBlockClassPath {
449 <        my $self=shift;
450 <        my $name=shift;
451 <        my $hashref=shift;
452 <
453 <        $self->{switch}->checktag($name, $hashref, 'path');
454 <        $self->{BlockClassPath}=$self->{BlockClassPath}.":".$$hashref{path};
455 <        $self->_CutBlock($$hashref{path});
456 < }
457 <
458 < sub BlockClassPath {
459 <        my $self=shift;
460 <        return $self->{BlockClassPath};
461 < }
462 <
463 < sub export_start_export {
464 <        my $self=shift;
465 <        my $name=shift;
466 <        my $hashref=shift;
467 <
468 <        $self->{switch}->opengroup("__export");
469 < }
470 <
471 < sub export_start {
472 <        my $self=shift;
473 <        my $name=shift;
474 <        my $hashref=shift;
475 <
476 <        $self->{switch}->opengroup("__export");
477 <        if ( exists $$hashref{autoexport} ) {
478 <          print GNUmakefile "scram_autoexport=".$$hashref{autoexport}."\n";
479 <          if ( $$hashref{autoexport}=~/true/ ) {
480 <           $self->{switch}->allowgroup("__export","makebuild");
481 <          }
482 <          else {
483 <           $self->{switch}->disallowgroup("__export","makebuild");
484 <          }
485 <        }
486 <        # -- allow default setting from other makefiles
487 <        print GNUmakefile "ifeq (\$(scram_autoexport),true)\n";
488 < }
489 <
490 < sub export_end_export {
491 <        my $self=shift;
492 <        $self->{switch}->closegroup("__export");
493 < }
494 <
495 < sub export_end {
496 <        my $self=shift;
497 <        $self->{switch}->closegroup("__export");
498 <        print GNUmakefile "endif\n";
499 < }
500 <
501 < #
502 < # Standard lib tag
503 < #
504 < sub lib_start {
505 <        my $self=shift;
506 <        my $name=shift;
507 <        my $hashref=shift;
508 <
509 <        $self->{switch}->checktag($name, $hashref, 'name');
510 <        if ( $self->{Arch} ) {
511 <           print GNUmakefile "lib+=$$hashref{name}\n";
512 <        }
513 < }
514 <
515 < #
516 < # libtype specification
517 < #
518 < sub LibType_Start {
519 <        my $self=shift;
520 <        my $name=shift;
521 <        my $hashref=shift;
522 <
523 <        if ( $self->{Arch} ) {
524 <        if ( defined $self->{libtype_conext} ) {
525 <          $self->{switch}->parseerror("<$name> tag cannot be specified".
526 <                " without a </$name> tag to close previous context");
527 <        }
528 <        else {
529 <        $self->{libtype_conext}=1;
530 <        $self->{switch}->checktag($name, $hashref, 'type');
531 <        
532 <        print GNUmakefile "# Specify Library Type\n";
533 <        print GNUmakefile "DefaultLibsOff=yes\n";
534 <        if ( $$hashref{'type'}=~/^archive/i ) {
535 <          print GNUmakefile "LibArchive=true\n";
536 <        }
537 <        elsif ($$hashref{'type'}=~/debug_archive/i ) {
538 <          print GNUmakefile "LibDebugArchive=true\n";
539 <        }
540 <        elsif ($$hashref{'type'}=~/debug_shared/i ) {
541 <          print GNUmakefile "LibDebugShared=true\n";
542 <        }
543 <        elsif ($$hashref{'type'}=~/shared/i ) {
544 <          print GNUmakefile 'LibShared=true'."\n";
545 <        }
546 <        print GNUmakefile "\n";
547 <        }
548 <        }
549 < }
550 <
551 < sub LibType_text {
552 <        my $self=shift;
553 <        my $name=shift;
554 <        my $string=shift;
555 <
556 <        if ( $self->{Arch} ) {
557 <          $string=~s/\n/ /g;
558 <          print GNUmakefile "libmsg::\n\t\@echo Library info: ";
559 <          print GNUmakefile $string;
560 <          print GNUmakefile "\n";
561 <        }
562 < }
563 <
564 < sub LibType_end {
565 <        my $self=shift;
566 <        my $name=shift;
567 <
568 <        undef $self->{libtype_conext};
569 < }
570 <
571 < sub Environment_start {
572 <        my $self=shift;
573 <        my $name=shift;
574 <        my $hashref=shift;
575 <
576 <        if ( $self->{Arch} ) {
577 <          $self->{envnum}++;
578 <
579 <          # open a new Environment File
580 <          my $envfile=$self->{localtop}."/$ENV{INTwork}/$self->{path}/Env_".
581 <                $self->{envnum}.".mk";
582 <          use FileHandle;
583 <          my $fh=FileHandle->new();
584 <          open ($fh,">$envfile") or die "Unable to open file $envfile \n$!\n";
585 <          push @{$self->{filehandlestack}}, $fh;
586 <          *GNUmakefile=$fh;
587 <
588 <          # include the approprate environment file
589 <          if ( $self->{envlevel} == 0 ) {
590 <             print GNUmakefile "include $self->{localtop}/$ENV{INTwork}/".
591 <                $self->{path}."/BuildFile.mk\n";
592 <          }
593 <          else {
594 <             print GNUmakefile "include $self->{localtop}/$ENV{INTwork}/".
595 <                $self->{path}."/Env_".$self->{Envlevels}[$self->{envlevel}].".mk\n";
596 <          }
597 <          $self->{envlevel}++;
598 <          $self->{Envlevels}[$self->{envlevel}]=$self->{envnum};
599 <          $self->{currentenv}="$self->{localtop}/$ENV{INTwork}/$self->{path}/Env_$self->{envnum}.mk";
600 <        }
601 < }
602 <
603 < sub Environment_end {
604 <        my $self=shift;
605 <        my $fd;
606 <
607 <        if ( $self->{Arch} ) {
608 <          $self->{envlevel}--;
609 <          if ( $self->{envlevel} < 0 ) {
610 <            print "Too many </Environent> Tags on $self->{switch}->line()\n";
611 <            exit 1;
612 <          }
613 <          close GNUmakefile;
614 <          # restore the last filehandle
615 <          $fd=pop @{$self->{filehandlestack}};
616 <          close $fd;
617 <          *GNUmakefile=$self->{filehandlestack}[$#{$self->{filehandlestack}}];
618 <          if ( $self->{envlevel} < 1 ) {
619 <            $self->{currentenv}="$self->{localtop}/$ENV{INTwork}/$self->{path}/".
620 <                        "BuildFile.mk";
621 <          }
622 <          else {
623 <            $self->{currentenv}=
624 <             $self->{localtop}."/$ENV{INTwork}/$self->{path}/Env_".
625 <                $self->{Envlevels}[$self->{envlevel}];
626 <          }
627 <        }
628 < }
629 <
630 < sub Store_start {
631 <        my $self=shift;
632 <        my $name=shift;
633 <        my $hashref=shift;
634 <
635 <        if ( $self->{Arch} ) {
636 <          $self->{switch}->checktag( $name, $hashref, 'name' );
637 <
638 <          # -- store creation
639 <          my $dir=$$hashref{'name'};
640 <          AddDir::adddir($self->{localtop}."/".$dir);
641 <          if ( exists $$hashref{'type'} ) {
642 <            # -- architecture specific store
643 <            if ( $$hashref{'type'}=~/^arch/i ) {
644 <                $dir=$dir."/".$ENV{SCRAM_ARCH};
645 <                AddDir::adddir($self->{localtop}."/".$dir);
646 <            }
647 <            else {
648 <                $self->parseerror("Unknown type in <$name> tag");
649 <            }
650 <          }
651 <
652 <          # -- set make variables for the store
653 <          print GNUmakefile "SCRAMSTORENAME_".$$hashref{'name'}.":=".$dir."\n";
654 <          print GNUmakefile "SCRAMSTORE_".$$hashref{'name'}.":=".
655 <                                        $self->{localtop}."/".$dir."\n";
656 <          print GNUmakefile "VPATH+=".$self->{localtop}
657 <                        ."/".$dir.":".$self->{releasetop}."/".$dir."\n";
1110 <        }
1111 < }
332 >      }
333 >   return "";
334 >   }
335 >
336 > sub bin()
337 >   {
338 >   my ($object,$name,%attributes) = @_;
339 >   $self->pushlevel(\%attributes);# Set nested to 1;
340 >   }
341 >
342 > sub bin_()
343 >   {
344 >   # Need unique name for the binary (always use name of product). Either use "name"
345 >   # given, or use "file" value minus the ending:
346 >   if (exists ($self->{id}->{'name'}))
347 >      {
348 >      $name = $self->{id}->{'name'};
349 >      }
350 >   else
351 >      {
352 >      ($name) = ($self->{id}->{'file'} =~ /(.*)?\..*$/);
353 >      }
354 >
355 >   # Store the data:
356 >   $self->productcollector($name,'bin','BIN');
357 >   $self->poplevel();
358 >   }
359 >
360 > sub module()
361 >   {
362 >   my ($object,$name,%attributes) = @_;
363 >   $self->pushlevel(\%attributes);# Set nested to 1;
364 >   }
365 >
366 > sub module_()
367 >   {
368 >   # Need unique name for the module (always use name of product). Either use "name"
369 >   # given, or use "file" value minus the ending:
370 >   if (exists ($self->{id}->{'name'}))
371 >      {
372 >      $name = $self->{id}->{'name'};
373 >      }
374 >   else
375 >      {
376 >      ($name) = ($self->{id}->{'file'} =~ /(.*)?\..*$/);
377 >      }
378 >
379 >   # Store the data:
380 >   $self->productcollector($name,'mod','MODULE');
381 >   $self->poplevel();
382 >   }
383 >
384 > sub application()
385 >   {
386 >   my ($object,$name,%attributes) = @_;
387 >   $self->pushlevel(\%attributes);# Set nested to 1;
388 >   }
389 >
390 > sub application_()
391 >   {
392 >   # Need unique name for the application (always use name of product). Either use "name"
393 >   # given, or use "file" value minus the ending:
394 >   if (exists ($self->{id}->{'name'}))
395 >      {
396 >      $name = $self->{id}->{'name'};
397 >      }
398 >   else
399 >      {
400 >      ($name) = ($self->{id}->{'file'} =~ /(.*)?\..*$/);
401 >      }
402 >
403 >   # Store the data:
404 >   $self->productcollector($name,'app','APPLICATION');
405 >   $self->poplevel();
406 >   }
407 >
408 > sub library()
409 >   {
410 >   my ($object,$name,%attributes) = @_;
411 >   $self->pushlevel(\%attributes);# Set nested to 1;
412 >   }
413 >
414 > sub library_()
415 >   {
416 >   # Need unique name for the library (always use name of product). Either use "name"
417 >   # given, or use "file" value minus the ending:
418 >   if (exists ($self->{id}->{'name'}))
419 >      {
420 >      $name = $self->{id}->{'name'};
421 >      }
422 >   else
423 >      {
424 >      ($name) = ($self->{id}->{'file'} =~ /(.*)?\..*$/);
425 >      }
426 >
427 >   # Store the data:
428 >   $self->productcollector($name,'lib','LIBRARY');
429 >   $self->poplevel();
430 >   }
431 >
432 > sub plugin()
433 >   {
434 >   my ($object,$name,%attributes) = @_;
435 >   $self->pushlevel(\%attributes);# Set nested to 1;
436 >   }
437 >
438 > sub plugin_()
439 >   {
440 >   # Need unique name for the plugin (always use name of product). Either use "name"
441 >   # given, or use "file" value minus the ending:
442 >   if (exists ($self->{id}->{'name'}))
443 >      {
444 >      $name = $self->{id}->{'name'};
445 >      }
446 >   else
447 >      {
448 >      ($name) = ($self->{id}->{'file'} =~ /(.*)?\..*$/);
449 >      }
450 >
451 >   # Store the data:
452 >   $self->productcollector($name,'plugin','PLUGIN');
453 >   $self->poplevel();
454 >   }
455 >
456 > sub unittest()
457 >   {
458 >   my ($object,$name,%attributes) = @_;
459 >   $self->pushlevel(\%attributes);# Set nested to 1;
460 >   }
461 >
462 > sub unittest_()
463 >   {
464 >   # Need unique name for the unittest (always use name of product). Either use "name"
465 >   # given, or use "file" value minus the ending:
466 >   if (exists ($self->{id}->{'name'}))
467 >      {
468 >      $name = $self->{id}->{'name'};
469 >      }
470 >   else
471 >      {
472 >      ($name) = ($self->{id}->{'file'} =~ /(.*)?\..*$/);
473 >      }
474 >
475 >   # Store the data:
476 >   $self->productcollector($name,'unittest','unittest');
477 >   $self->poplevel();
478 >   }
479 >
480 > sub productcollector()
481 >   {
482 >   my $self=shift;
483 >   my ($name,$typeshort,$typefull)=@_;
484 >   # Create a new Product object for storage of data:
485 >   use BuildSystem::Product;
486 >   my $product = BuildSystem::Product->new();
487 >   # Store the name:
488 >   $product->name($name);
489 >   $product->type($typeshort);
490 >   # Store the files. Take the BuildFile path as the initial path for
491 >   # expanding source file globs:
492 >   $product->_files($self->{id}->{'file'},[ $self->{scramdoc}->filetoparse() ]);
493 >   # Store the data content:
494 >   $product->_data($self->{tagcontent});
495 >   # And store in a hash (all build products in same place):
496 >   $self->{content}->{BUILDPRODUCTS}->{$typefull}->{$name} = $product;
497 >   }
498 >
499 > sub pushlevel
500 >   {
501 >   my $self = shift;
502 >   my ($info)=@_;
503 >  
504 >   $self->{id} = $info if (defined $info);
505 >   $self->{nested} = 1;
506 >   $self->{tagcontent}={};
507 >   }
508 >
509 > sub poplevel
510 >   {
511 >   my $self = shift;
512 >   delete $self->{id};
513 >   delete $self->{nested};
514 >   delete $self->{tagcontent};
515 >   }
516 >
517 > sub dependencies()
518 >   {
519 >   my $self=shift;
520 >   # Make a copy of the variable so that
521 >   # we don't have a DEPENDENCIES entry in RAWDATA:
522 >   my %DEPS=%{$self->{DEPENDENCIES}};
523 >   delete $self->{DEPENDENCIES};
524 >   return \%DEPS;
525 >   }
526 >
527 > sub skippeddirs()
528 >   {
529 >   my $self=shift;
530 >   my ($here)=@_;
531 >   my $skipped;
532 >
533 >   if ($self->{content}->{SKIPPEDDIRS}->[0] == 1)
534 >      {
535 >      $skipped = [ @{$self->{content}->{SKIPPEDDIRS}} ];
536 >      delete $self->{content}->{SKIPPEDDIRS};
537 >      }
538 >  
539 >   delete $self->{content}->{SKIPPEDDIRS};
540 >   return $skipped;
541 >   }
542 >
543 > sub hasexport()
544 >   {
545 >   my $self=shift;
546 >   # Check to see if there is a valid export block:
547 >   my $nkeys = $self->exporteddatatypes();
548 >   $nkeys > 0 ? return 1 : return 0;
549 >   }
550 >
551 > sub has()
552 >   {
553 >   my $self=shift;
554 >   my ($datatype)=@_;  
555 >   (exists ($self->{content}->{$datatype})) ? return 1 : return 0;
556 >   }
557 >
558 > sub exported()
559 >   {
560 >   my $self=shift;
561 >   # Return a hash. Keys are type of data provided:
562 >   return ($self->{content}->{EXPORT});
563 >   }
564 >
565 > sub exporteddatatypes()
566 >   {
567 >   my $self=shift;
568 >   # Return exported data types:
569 >   return keys %{$self->{content}->{EXPORT}};
570 >   }
571 >
572 > sub defined_group()
573 >   {
574 >   my $self=shift;
575 >
576 >   if (exists($self->{content}->{DEFINED_GROUP}))
577 >      {  
578 >      # Return a list of keys (group names) for defined groups:
579 >      return [ keys %{$self->{content}->{DEFINED_GROUP}} ];
580 >      }
581 >   else
582 >      {
583 >      return 0;
584 >      }
585 >   }
586 >
587 > sub dataforgroup()
588 >   {
589 >   my $self=shift;
590 >   my ($groupname)=@_;
591 >   # Return hash containing data for defined group
592 >   # $groupname or return undef:
593 >   return $self->{content}->{DEFINED_GROUP}->{$groupname};
594 >   }
595 >
596 > sub buildproducts()
597 >   {
598 >   my $self=shift;
599 >   # Returns hash of build products and their data:
600 >   return $self->{content}->{BUILDPRODUCTS};
601 >   }
602 >
603 > sub values()
604 >   {
605 >   my $self=shift;
606 >   my ($type)=@_;
607 >   # Get a list of values from known types
608 >   return $self->{content}->{BUILDPRODUCTS}->{$type};
609 >   }
610 >
611 > sub basic_tags()
612 >   {
613 >   my $self=shift;
614 >   my $datatags=[];
615 >   my $buildtags=[ qw(BIN LIBRARY APPLICATION MODULE PLUGIN BUILDPRODUCTS) ];
616 >   my $skiptags=[ qw(DEFINED_GROUP ARCH EXPORT GROUP USE CLASSPATH) ];
617 >   my $otherskiptags=[ qw( SKIPPEDDIRS ) ];
618 >   my @all_skip_tags;
619 >  
620 >   push(@all_skip_tags,@$skiptags,@$buildtags,@$otherskiptags);
621 >
622 >   foreach my $t (keys %{$self->{content}})
623 >      {
624 >      push(@$datatags,$t),if (! grep($t eq $_, @all_skip_tags));
625 >      }
626 >   return @{$datatags};
627 >   }
628 >
629 > sub clean()
630 >   {
631 >   my $self=shift;
632 >   my (@tags) = @_;
633 >
634 >   # Delete some useless entries:
635 >   delete $self->{makefilecontent};
636 >   delete $self->{simpledoc};
637 >   delete $self->{id};
638 >   delete $self->{tagcontent};
639 >   delete $self->{nested};
640 >
641 >   delete $self->{DEPENDENCIES};
642 >  
643 >   map
644 >      {
645 >      delete $self->{content}->{$_} if (exists($self->{content}->{$_}));
646 >      } @tags;
647 >  
648 >   return $self;
649 >   }
650 >
651 > sub AUTOLOAD()
652 >   {
653 >   my ($xmlparser,$name,%attributes)=@_;
654 >   return if $AUTOLOAD =~ /::DESTROY$/;
655 >   my $name=$AUTOLOAD;
656 >   $name =~ s/.*://;
657 >   }
658  
659 + 1;

Diff Legend

Removed lines
+ Added lines
< Changed lines
> Changed lines