2 // TargetBatchingImpl.cs: Class that implements Target Batching Algorithm from the wiki.
5 // Marek Sieradzki (marek.sieradzki@gmail.com)
6 // Ankit Jain (jankit@novell.com)
8 // (C) 2005 Marek Sieradzki
9 // Copyright 2008 Novell, Inc (http://www.novell.com)
10 // Copyright 2009 Novell, Inc (http://www.novell.com)
12 // Permission is hereby granted, free of charge, to any person obtaining
13 // a copy of this software and associated documentation files (the
14 // "Software"), to deal in the Software without restriction, including
15 // without limitation the rights to use, copy, modify, merge, publish,
16 // distribute, sublicense, and/or sell copies of the Software, and to
17 // permit persons to whom the Software is furnished to do so, subject to
18 // the following conditions:
20 // The above copyright notice and this permission notice shall be
21 // included in all copies or substantial portions of the Software.
23 // THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
24 // EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
25 // MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
26 // NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE
27 // LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
28 // OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
29 // WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
35 using System
.Collections
.Generic
;
38 using Microsoft
.Build
.Framework
;
40 namespace Microsoft
.Build
.BuildEngine
{
42 internal class TargetBatchingImpl
: BatchingImplBase
48 public TargetBatchingImpl (Project project
, XmlElement targetElement
)
51 if (targetElement
== null)
52 throw new ArgumentNullException ("targetElement");
54 inputs
= targetElement
.GetAttribute ("Inputs");
55 outputs
= targetElement
.GetAttribute ("Outputs");
56 name
= targetElement
.GetAttribute ("Name");
59 public bool Build (Target target
, out bool executeOnErrors
)
61 executeOnErrors
= false;
64 if (!BuildTargetNeeded (out reason
)) {
65 LogTargetStarted (target
);
66 LogTargetSkipped (target
, reason
);
67 LogTargetFinished (target
, true);
71 if (!String
.IsNullOrEmpty (reason
))
72 target
.Engine
.LogMessage (MessageImportance
.Low
, reason
);
76 ParseTargetAttributes (target
);
77 BatchAndPrepareBuckets ();
78 return Run (target
, out executeOnErrors
);
80 consumedItemsByName
= null;
81 consumedMetadataReferences
= null;
82 consumedQMetadataReferences
= null;
83 consumedUQMetadataReferences
= null;
84 batchedItemsByName
= null;
85 commonItemsByName
= null;
89 bool Run (Target target
, out bool executeOnErrors
)
91 executeOnErrors
= false;
92 if (buckets
.Count
> 0) {
93 foreach (Dictionary
<string, BuildItemGroup
> bucket
in buckets
)
94 if (!RunTargetWithBucket (bucket
, target
, out executeOnErrors
))
99 return RunTargetWithBucket (null, target
, out executeOnErrors
);
103 bool RunTargetWithBucket (Dictionary
<string, BuildItemGroup
> bucket
, Target target
, out bool executeOnErrors
)
105 bool target_result
= true;
106 executeOnErrors
= false;
108 LogTargetStarted (target
);
110 project
.PushBatch (bucket
, commonItemsByName
);
113 if (!BuildTargetNeeded (out reason
)) {
114 LogTargetSkipped (target
, reason
);
118 if (!String
.IsNullOrEmpty (reason
))
119 target
.Engine
.LogMessage (MessageImportance
.Low
, reason
);
121 for (int i
= 0; i
< target
.BuildTasks
.Count
; i
++) {
122 //FIXME: parsing attributes repeatedly
123 BuildTask bt
= target
.BuildTasks
[i
];
125 TaskBatchingImpl batchingImpl
= new TaskBatchingImpl (project
);
126 bool task_result
= batchingImpl
.Build (bt
, out executeOnErrors
);
130 // task failed, if ContinueOnError,
131 // ignore failed state for target
132 target_result
= bt
.ContinueOnError
;
134 if (!bt
.ContinueOnError
) {
135 executeOnErrors
= true;
143 LogTargetFinished (target
, target_result
);
146 return target_result
;
149 // Parse target's Input and Output attributes to get list of referenced
150 // metadata and items to determine batching
151 void ParseTargetAttributes (Target target
)
153 if (!String
.IsNullOrEmpty (inputs
))
154 ParseAttribute (inputs
);
156 if (!String
.IsNullOrEmpty (outputs
))
157 ParseAttribute (outputs
);
160 bool BuildTargetNeeded (out string reason
)
162 reason
= String
.Empty
;
163 ITaskItem
[] inputFiles
;
164 ITaskItem
[] outputFiles
;
165 DateTime youngestInput
, oldestOutput
;
167 if (String
.IsNullOrEmpty (inputs
.Trim ()))
170 if (String
.IsNullOrEmpty (outputs
.Trim ())) {
171 project
.ParentEngine
.LogError ("Target {0} has inputs but no outputs specified.", name
);
175 Expression e
= new Expression ();
176 e
.Parse (inputs
, ParseOptions
.AllowItemsMetadataAndSplit
);
177 inputFiles
= (ITaskItem
[]) e
.ConvertTo (project
, typeof (ITaskItem
[]), ExpressionOptions
.ExpandItemRefs
);
179 e
= new Expression ();
180 e
.Parse (outputs
, ParseOptions
.AllowItemsMetadataAndSplit
);
181 outputFiles
= (ITaskItem
[]) e
.ConvertTo (project
, typeof (ITaskItem
[]), ExpressionOptions
.ExpandItemRefs
);
183 if (outputFiles
== null || outputFiles
.Length
== 0) {
184 reason
= String
.Format ("No output files were specified for target {0}, skipping.", name
);
188 if (inputFiles
== null || inputFiles
.Length
== 0) {
189 reason
= String
.Format ("No input files were specified for target {0}, skipping.", name
);
193 youngestInput
= DateTime
.MinValue
;
194 oldestOutput
= DateTime
.MaxValue
;
196 string youngestInputFile
, oldestOutputFile
;
197 youngestInputFile
= oldestOutputFile
= String
.Empty
;
198 foreach (ITaskItem item
in inputFiles
) {
199 string file
= item
.ItemSpec
.Trim ();
200 if (file
.Length
== 0)
203 if (!File
.Exists (file
)) {
204 reason
= String
.Format ("Target {0} needs to be built as input file '{1}' does not exist.", name
, file
);
208 DateTime lastWriteTime
= File
.GetLastWriteTime (file
);
209 if (lastWriteTime
> youngestInput
) {
210 youngestInput
= lastWriteTime
;
211 youngestInputFile
= file
;
215 foreach (ITaskItem item
in outputFiles
) {
216 string file
= item
.ItemSpec
.Trim ();
217 if (file
.Length
== 0)
220 if (!File
.Exists (file
)) {
221 reason
= String
.Format ("Target {0} needs to be built as output file '{1}' does not exist.", name
, file
);
225 DateTime lastWriteTime
= File
.GetLastWriteTime (file
);
226 if (lastWriteTime
< oldestOutput
) {
227 oldestOutput
= lastWriteTime
;
228 oldestOutputFile
= file
;
232 if (youngestInput
> oldestOutput
) {
233 reason
= String
.Format ("Target {0} needs to be built as input file '{1}' is newer than output file '{2}'",
234 name
, youngestInputFile
, oldestOutputFile
);
241 void LogTargetSkipped (Target target
, string reason
)
243 BuildMessageEventArgs bmea
;
244 bmea
= new BuildMessageEventArgs (reason
?? String
.Format ("Skipping target \"{0}\" because its outputs are up-to-date.", target
.Name
),
245 null, "MSBuild", MessageImportance
.Normal
);
246 target
.Engine
.EventSource
.FireMessageRaised (this, bmea
);
249 void LogTargetStarted (Target target
)
251 TargetStartedEventArgs tsea
;
252 string projectFile
= project
.FullFileName
;
253 tsea
= new TargetStartedEventArgs (String
.Format ("Target {0} started.", target
.Name
), null,
254 target
.Name
, projectFile
, target
.TargetFile
);
255 target
.Engine
.EventSource
.FireTargetStarted (this, tsea
);
258 void LogTargetFinished (Target target
, bool succeeded
)
260 TargetFinishedEventArgs tfea
;
261 string projectFile
= project
.FullFileName
;
262 tfea
= new TargetFinishedEventArgs (String
.Format ("Target {0} finished.", target
.Name
), null,
263 target
.Name
, projectFile
, target
.TargetFile
, succeeded
);
264 target
.Engine
.EventSource
.FireTargetFinished (this, tfea
);