PSCF v1.2
make.py
1'''! Python scripts used by the PSCF makefile build system. '''
2
3import os
4import os.path
5import glob
6from string import *
7
8from pscfpp.file import *
9from pscfpp.text import *
10
11
30def createDependencyFileCpp(processor, options, cfile, srcdir, blddir, extraDependencies=''):
31
32 # Isolate base source file name
33 base = os.path.basename(cfile)
34 groups = base.split('.')
35 base = groups[0]
36
37 # Calculate source file directory path relative to srcdir
38 abspath = os.path.abspath(cfile)
39 relpath = os.path.relpath(abspath, srcdir)
40 reldir = os.path.dirname(relpath)
41
42 # Construct paths to subdirectories of build and src directories
43 if (reldir != '.' and reldir != ''):
44 blddir = os.path.normpath(os.path.join(blddir, reldir))
45 srcdir = os.path.normpath(os.path.join(srcdir, reldir))
46
47 # Construct paths to dependency files
48 # pfile - temporary file created by the compiler (erased)
49 # dfile - final dependnecy file
50 pfile = os.path.normpath(os.path.join(srcdir, base)) + '.p'
51 dfile = os.path.normpath(os.path.join(blddir, base)) + '.d'
52
53 # Create and execute compiler command to calculate dependencies
54 command = processor + ' '
55 command += pfile + ' '
56 command += options + ' '
57 command += cfile
58 #print('\n' + command + '\n')
59 os.system(command)
60
61 #Edit dependency file
62 editDepend(pfile, dfile, blddir, extraDependencies)
63 os.remove(pfile)
64
65
66
85def createDependencyFileCuda(processor, options, cfile, srcdir, blddir, extraDependencies=''):
86
87 # Store unmodified srcdir
88 srcroot = srcdir
89
90 # Isolate base source file name
91 base = os.path.basename(cfile)
92 groups = base.split('.')
93 base = groups[0]
94
95 # Calculate source file directory path relative to srcdir
96 abspath = os.path.abspath(cfile)
97 relpath = os.path.relpath(abspath, srcdir)
98 reldir = os.path.dirname(relpath)
99
100 # Construct paths to subdirectories of build and src directories
101 if (reldir != '.' and reldir != ''):
102 blddir = os.path.normpath(os.path.join(blddir, reldir))
103 srcdir = os.path.normpath(os.path.join(srcdir, reldir))
104
105 # Construct paths to dependency files
106 # pfile - temporary file created by the compiler (erased)
107 # dfile - final dependnecy file
108 pfile = os.path.normpath(os.path.join(srcdir, base)) + '.p'
109 dfile = os.path.normpath(os.path.join(blddir, base)) + '.d'
110
111 # Create and execute compiler command to calculate dependencies
112 command = processor + ' '
113 command += options + ' '
114 command += cfile
115 command += ' > ' + pfile
116 os.system(command)
117
118 # Edit dependency file and remove temporary pfile
119 editDependLocal(pfile, dfile, blddir, srcroot, extraDependencies)
120 os.remove(pfile)
121
122
123
135def editDepend(pfile, dfile, blddir, extraDependencies):
136 file = open(pfile, 'r')
137 lines = file.readlines()
138 file.close()
139
140 # Extract target from first line
141 groups = lines[0].split(":")
142 target = groups[0]
143 lines[0] = groups[1]
144
145 # Replace target by file in build directory
146 base = os.path.basename(target)
147 target = os.path.normpath(os.path.join(blddir, base))
148
149 # Replace dependencies by absolute paths
150 text = Wrapper('\\\n', 8)
151 for i in range(len(lines)):
152 line = lines[i]
153 if line[-1] == '\n':
154 line = line[:-1]
155 if line[-1] == '\\':
156 line = line[:-1]
157 lines[i] = line
158 if i == 0:
159 text.append(target + ': ')
160 deps = line.split()
161 for dep in deps:
162 path = os.path.abspath(dep)
163 text.append(path)
164
165 # Process extraDependencies (if any)
166 if extraDependencies:
167 deps = extraDependencies.split()
168 for dep in deps:
169 path = os.path.abspath(dep)
170 text.append(path)
171
172 file = open(dfile, 'w')
173 file.write(str(text))
174 file.close()
175
176
189def editDependLocal(pfile, dfile, blddir, srcroot, extraDependencies):
190 file = open(pfile, 'r')
191 lines = file.readlines()
192 file.close()
193
194 # Extract target from first line
195 groups = lines[0].split(":")
196 target = groups[0]
197 lines[0] = groups[1]
198
199 # Replace target by file in build directory
200 base = os.path.basename(target)
201 target = os.path.normpath(os.path.join(blddir, base))
202
203 # Replace local dependencies by absolute paths
204 text = Wrapper('\\\n', 8)
205 for i in range(len(lines)):
206 line = lines[i]
207 if line[-1] == '\n':
208 line = line[:-1]
209 if line[-1] == '\\':
210 line = line[:-1]
211 lines[i] = line
212 if i == 0:
213 text.append(target + ': ')
214 deps = line.split()
215 for dep in deps:
216 pair = [srcroot, dep]
217 if (os.path.commonprefix(pair) == srcroot):
218 path = os.path.abspath(dep)
219 text.append(path)
220
221 # Process extraDependencies (if any)
222 if extraDependencies:
223 deps = extraDependencies.split()
224 for dep in deps:
225 path = os.path.abspath(dep)
226 text.append(path)
227
228 file = open(dfile, 'w')
229 file.write(str(text))
230 file.close()
231
232
237
238
247 def __init__(self, path = '.', pathFromSrc = '.', pathToSrc ='.', parent = None, isTest=False):
248
249 self.path = path # path from working directory
250 self.pathFromSrc = pathFromSrc # path from $(SRC_DIR)
251 self.pathToSrc = pathToSrc # path to $(SRC_DIR)
252 self.parent = parent
253 self.isTest = isTest
254 self.hasLib = False
255 self.globalInclude = ''
256 self.defines = ''
257
258 if parent:
259 self.root = parent.root
260 self.generationgeneration = parent.generation + 1
261 else:
262 self.root = self
263 self.generationgeneration = 0
264
265 # Parse directory path
266 if self.pathFromSrc == '.':
267 self.dirparts = []
268 self.dirname = 'SRC'
269 else:
270 self.dirparts = self.pathFromSrc.split(os.sep)
271 self.dirname = '_'.join(self.dirparts)
272
273 # Read dir.txt file
274 dirFilePath = self.makePath('dir.txt')
275 if os.path.exists(dirFilePath):
276 file = open(dirFilePath,'r')
277 self.dirList = readLabelledList(file, 'subDirectories')
278 file.close()
279 else:
280 self.dirList = []
281
282 # Create a child MakeMaker for each directory
283 self.dirs = []
284 for dir in self.dirList:
285 path = self.makePath(dir)
286 if os.path.isdir(path):
287 if self.pathFromSrc == '.':
288 pathFromSrc = dir
289 pathToSrc = '..'
290 else:
291 pathFromSrc = self.pathFromSrc + os.sep + dir
292 pathToSrc = '..' + os.sep + self.pathToSrc
293 if dir == 'tests':
294 maker = MakeMaker(path, pathFromSrc, pathToSrc, self, True)
295 else:
296 maker = MakeMaker(path, pathFromSrc, pathToSrc, self, self.isTest)
297 self.dirs.append(maker)
298
299 def setGlobalInclude(self, text):
300 self.globalInclude = text
301 for dir in self.dirs:
302 dir.setGlobalInclude(text)
303
304 def setDefines(self, text):
305 self.defines = text
306 for dir in self.dirs:
307 dir.setDefines(text)
308
309 def addLibrary(self, libName, libObjs):
310 self.hasLib = True
311 self.libName = libName
312 self.libObjs = libObjs
313
314 def setLinkObjs(self, text):
315 self.linkObjs = text
316 for dir in self.dirs:
317 dir.setLinkObjs(text)
318
319 def makeInclude(self, base):
320 ''' Make include line suitable for a makefile'''
321 if self.pathFromSrc == '.':
322 return 'include $(SRC_DIR)' + os.sep + base + '\n'
323 else:
324 return 'include $(SRC_DIR)' + os.sep + self.pathFromSrc + os.sep + base + '\n'
325
326 def srcSuffix(self):
327 ''' Return suffix for source files: cpp or cc '''
328 if (self.isTest):
329 return 'cc'
330 else:
331 return 'cpp'
332
333 def find(self):
334 '''
335 Find all header and source files in this directory.
336 Note: Does not recursively descend into subdirectories
337 '''
338 # Find source files
339 self.srcssrcs = []
340 self.hasSrcs = False
341 for name in os.listdir(self.path):
342 if name != '.':
343 path = self.makePath(name)
344 if os.path.isfile(path):
345 base = os.path.basename(path)
346 groups = base.split('.')
347 if len(groups) == 2:
348 base = groups[0]
349 suffix = groups[1]
350 if suffix == self.srcSuffix():
351 path = self.makePathFromSrc(base)
352 self.srcssrcs.append(base)
353 self.hasSrcs = True
354
355 # Find *.h header files
356 self.hdrs = []
357 self.hasHdrs = False
358 for name in os.listdir(self.path):
359 if name != '.':
360 path = self.makePath(name)
361 if os.path.isfile(path):
362 base = os.path.basename(path)
363 groups = base.split('.')
364 if len(groups) == 2:
365 base = groups[0]
366 suffix = groups[1]
367 if suffix == 'h':
368 if base not in self.srcssrcs:
369 self.hdrs.append(base)
370 self.hasHdrs = True
371
372 def make(self):
373
374 # Find all the source and header files in this directory
375 self.find()
376
377 # Recursively descend into subdirectories
378 for dir in self.dirs:
379 dir.make()
380
381 # ---------------------------------------------------------------------
382 # Open and write sources.mk file
383 file = open_w(self.makePath('sources.mk'))
384
385 # Include subdirectory sources.mk files
386 if len(self.dirs) > 0:
387 for dir in self.dirs:
388 basename = os.path.basename(dir.path)
389 if basename != 'tests':
390 varpath = basename + os.sep + 'sources.mk'
391 file.write(self.makeInclude(varpath))
392 file.write('\n')
393
394 # Construct prefix for aggregate names
395 if self.dirname == 'SRC':
396 prefix = ''
397 else:
398 prefix = self.dirname + '_'
399
400 wrapper = Wrapper('\\\n', 4)
401
402 # Write aggregate definition for SRCS
403 wrapper.clear()
404 wrapper.append(prefix + 'SRCS=')
405 for dir in self.dirs:
406 basename = os.path.basename(dir.path)
407 if basename != 'tests':
408 wrapper.append('$(' + dir.dirname + '_SRCS) ')
409 for base in self.srcssrcs:
410 name = '$(SRC_DIR)' + os.sep + self.makePathFromSrc(base)
411 name += '.' + self.srcSuffix() + ' '
412 wrapper.append(name)
413 file.write(str(wrapper))
414 file.write('\n\n')
415
416 # Write aggregate definition for OBJS
417 file.write(prefix + 'OBJS=$(')
418 file.write(prefix + 'SRCS:.' + self.srcSuffix() + '=.o)')
419 file.write('\n\n')
420 #file.write(prefix + 'DEPS=$(')
421 #file.write(prefix + 'SRCS:.' + self.srcSuffix() + '=.d)')
422 #file.write('\n\n')
423
424 # Add library target, if any
425 if self.hasLib:
426 file.write(self.libName + ': ' + self.libObjs + '\n')
427 file.write('\t$(AR) rcs ' + self.libName + ' ' + self.libObjs + '\n')
428 file.write('\n')
429
430 file.close()
431
432 # --------------------------------------------------------------------
433 # Write makefile
434
435 file = open(self.makePath('makefile'), 'w')
436 file.write('SRC_DIR_REL =' + self.pathToSrc + '\n\n')
437 file.write(self.globalInclude)
438 if self.isTest:
439 file.write('include $(SRC_DIR_REL)/test/sources.mk\n')
440 file.write('include sources.mk\n')
441 file.write('\n')
442 if self.dirname == 'SRC':
443 objs = 'OBJS'
444 else:
445 objs = self.dirname + '_OBJS'
446 targets = '$(' + objs + ')'
447 deps = '$(' + objs + ':.o=.d)'
448 if self.isTest:
449 for base in self.srcssrcs:
450 targets += ' ' + base
451 file.write('all: ' + targets)
452 if self.hasLib:
453 file.write(' ' + self.libName)
454 file.write('\n\n')
455 file.write('clean:\n\trm -f ' + targets + ' ' + deps)
456 if self.hasLib:
457 file.write(' ' + self.libName)
458 file.write('\n\n')
459 file.write('clean-deps:\n\trm -f ' + deps)
460 file.write('\n\n')
461 if self.isTest:
462 for base in self.srcssrcs:
463 file.write(base + ': ' + base + '.o ' + self.linkObjs + '\n')
464 file.write('\t$(CXX) $(LDFLAGS) $(INCLUDES) $(DEFINES)')
465 file.write(' -o ' + base + ' ' + base + '.o \\\n')
466 file.write('\t ' + self.linkObjs + '\n\n')
467 #file.write('-include $(' + objs + ':.o=.d)\n\n')
468 file.write('-include ' + deps + '\n\n')
469 file.close()
470
471 def makePath(self, string):
472 if self.path and self.path != '.':
473 return self.path + os.sep + string
474 else:
475 return string
476
477 def makePathFromSrc(self, string):
478 if self.pathFromSrc and self.pathFromSrc != '.':
479 return self.pathFromSrc + os.sep + string
480 else:
481 return string
482
483 def srcDirPath(self):
484 if self.generationgeneration == 0:
485 return self.path
486 else:
487 shifts = []
488 for i in range(self.generationgeneration):
489 shifts.append('..')
490 return os.sep.join(shifts)
491
492 def clear(self):
493 self.hdrs = []
494 self.srcssrcs = []
495 self.dirs = []
496
497 def filenames(self, pattern = '*', recursive = 1):
498 r = glob.glob( self.path + '/' + pattern)
499 if recursive:
500 for dir in self.dirs:
501 r.extend(dir.filenames(pattern))
502 return r
503
504 def __repr__(self):
505 r = []
506 r.append( self.path + '\n' )
507 for x in self.hdrs:
508 r.append( str(x) + '\n' )
509 for x in self.srcssrcs.keys() :
510 r.append( str(x) + '\n' )
511 for x in self.dirs:
512 r.append( str(x) )
513 return join(r,'')
514
515 def __str__(self):
516 return __repr__()
517
518 def ls(self):
519 for x in self.hdrs:
520 print(x)
521 for x in self.srcssrcs:
522 print(x)
523 for x in self.dirs:
524 print(x + os.sep)
525
Class to construct makefile system for a set of source files.
Definition make.py:236
find(self)
Find all header and source files in this directory.
Definition make.py:333
makeInclude(self, base)
Make include line suitable for a makefile.
Definition make.py:319
makePathFromSrc(self, string)
Definition make.py:477
__init__(self, path='.', pathFromSrc='.', pathToSrc='.', parent=None, isTest=False)
Constructor.
Definition make.py:247
srcSuffix(self)
Return suffix for source files: cpp or cc.
Definition make.py:326
makePath(self, string)
Definition make.py:471
Class to wrap line breaks.
Definition text.py:12
Utilities for manipulating files and paths.
Definition file.py:1
open_w(path)
Open file with specified path for writing, return file object.
Definition file.py:57
editDependLocal(pfile, dfile, blddir, srcroot, extraDependencies)
Edit the dependency file created for a CUDA file by the compiler.
Definition make.py:189
editDepend(pfile, dfile, blddir, extraDependencies)
Edit the dependency file created for a C/C++ file by the compiler.
Definition make.py:135
createDependencyFileCuda(processor, options, cfile, srcdir, blddir, extraDependencies='')
Create a *.d dependency file for a CUDA source file.
Definition make.py:85
createDependencyFileCpp(processor, options, cfile, srcdir, blddir, extraDependencies='')
Create a *.d dependency file for a C/C++ source file.
Definition make.py:30
Utilities for manipulating and searching text strings.
Definition text.py:1
readLabelledList(file, label)
Read line of form "label = string", in which string may contains spaces.
Definition text.py:263