mirror of
				https://github.com/python/cpython.git
				synced 2025-10-25 15:58:57 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			316 lines
		
	
	
	
		
			9.9 KiB
		
	
	
	
		
			Python
		
	
	
		
			Executable file
		
	
	
	
	
			
		
		
	
	
			316 lines
		
	
	
	
		
			9.9 KiB
		
	
	
	
		
			Python
		
	
	
		
			Executable file
		
	
	
	
	
| #! /usr/bin/env python3
 | |
| 
 | |
| # Perform massive identifier substitution on C source files.
 | |
| # This actually tokenizes the files (to some extent) so it can
 | |
| # avoid making substitutions inside strings or comments.
 | |
| # Inside strings, substitutions are never made; inside comments,
 | |
| # it is a user option (off by default).
 | |
| #
 | |
| # The substitutions are read from one or more files whose lines,
 | |
| # when not empty, after stripping comments starting with #,
 | |
| # must contain exactly two words separated by whitespace: the
 | |
| # old identifier and its replacement.
 | |
| #
 | |
| # The option -r reverses the sense of the substitutions (this may be
 | |
| # useful to undo a particular substitution).
 | |
| #
 | |
| # If the old identifier is prefixed with a '*' (with no intervening
 | |
| # whitespace), then it will not be substituted inside comments.
 | |
| #
 | |
| # Command line arguments are files or directories to be processed.
 | |
| # Directories are searched recursively for files whose name looks
 | |
| # like a C file (ends in .h or .c).  The special filename '-' means
 | |
| # operate in filter mode: read stdin, write stdout.
 | |
| #
 | |
| # Symbolic links are always ignored (except as explicit directory
 | |
| # arguments).
 | |
| #
 | |
| # The original files are kept as back-up with a "~" suffix.
 | |
| #
 | |
| # Changes made are reported to stdout in a diff-like format.
 | |
| #
 | |
| # NB: by changing only the function fixline() you can turn this
 | |
| # into a program for different changes to C source files; by
 | |
| # changing the function wanted() you can make a different selection of
 | |
| # files.
 | |
| 
 | |
| import sys
 | |
| import re
 | |
| import os
 | |
| from stat import *
 | |
| import getopt
 | |
| 
 | |
| err = sys.stderr.write
 | |
| dbg = err
 | |
| rep = sys.stdout.write
 | |
| 
 | |
| def usage():
 | |
|     progname = sys.argv[0]
 | |
|     err('Usage: ' + progname +
 | |
|               ' [-c] [-r] [-s file] ... file-or-directory ...\n')
 | |
|     err('\n')
 | |
|     err('-c           : substitute inside comments\n')
 | |
|     err('-r           : reverse direction for following -s options\n')
 | |
|     err('-s substfile : add a file of substitutions\n')
 | |
|     err('\n')
 | |
|     err('Each non-empty non-comment line in a substitution file must\n')
 | |
|     err('contain exactly two words: an identifier and its replacement.\n')
 | |
|     err('Comments start with a # character and end at end of line.\n')
 | |
|     err('If an identifier is preceded with a *, it is not substituted\n')
 | |
|     err('inside a comment even when -c is specified.\n')
 | |
| 
 | |
| def main():
 | |
|     try:
 | |
|         opts, args = getopt.getopt(sys.argv[1:], 'crs:')
 | |
|     except getopt.error as msg:
 | |
|         err('Options error: ' + str(msg) + '\n')
 | |
|         usage()
 | |
|         sys.exit(2)
 | |
|     bad = 0
 | |
|     if not args: # No arguments
 | |
|         usage()
 | |
|         sys.exit(2)
 | |
|     for opt, arg in opts:
 | |
|         if opt == '-c':
 | |
|             setdocomments()
 | |
|         if opt == '-r':
 | |
|             setreverse()
 | |
|         if opt == '-s':
 | |
|             addsubst(arg)
 | |
|     for arg in args:
 | |
|         if os.path.isdir(arg):
 | |
|             if recursedown(arg): bad = 1
 | |
|         elif os.path.islink(arg):
 | |
|             err(arg + ': will not process symbolic links\n')
 | |
|             bad = 1
 | |
|         else:
 | |
|             if fix(arg): bad = 1
 | |
|     sys.exit(bad)
 | |
| 
 | |
| # Change this regular expression to select a different set of files
 | |
| Wanted = r'^[a-zA-Z0-9_]+\.[ch]$'
 | |
| def wanted(name):
 | |
|     return re.match(Wanted, name)
 | |
| 
 | |
| def recursedown(dirname):
 | |
|     dbg('recursedown(%r)\n' % (dirname,))
 | |
|     bad = 0
 | |
|     try:
 | |
|         names = os.listdir(dirname)
 | |
|     except OSError as msg:
 | |
|         err(dirname + ': cannot list directory: ' + str(msg) + '\n')
 | |
|         return 1
 | |
|     names.sort()
 | |
|     subdirs = []
 | |
|     for name in names:
 | |
|         if name in (os.curdir, os.pardir): continue
 | |
|         fullname = os.path.join(dirname, name)
 | |
|         if os.path.islink(fullname): pass
 | |
|         elif os.path.isdir(fullname):
 | |
|             subdirs.append(fullname)
 | |
|         elif wanted(name):
 | |
|             if fix(fullname): bad = 1
 | |
|     for fullname in subdirs:
 | |
|         if recursedown(fullname): bad = 1
 | |
|     return bad
 | |
| 
 | |
| def fix(filename):
 | |
| ##  dbg('fix(%r)\n' % (filename,))
 | |
|     if filename == '-':
 | |
|         # Filter mode
 | |
|         f = sys.stdin
 | |
|         g = sys.stdout
 | |
|     else:
 | |
|         # File replacement mode
 | |
|         try:
 | |
|             f = open(filename, 'r')
 | |
|         except IOError as msg:
 | |
|             err(filename + ': cannot open: ' + str(msg) + '\n')
 | |
|             return 1
 | |
|         head, tail = os.path.split(filename)
 | |
|         tempname = os.path.join(head, '@' + tail)
 | |
|         g = None
 | |
|     # If we find a match, we rewind the file and start over but
 | |
|     # now copy everything to a temp file.
 | |
|     lineno = 0
 | |
|     initfixline()
 | |
|     while 1:
 | |
|         line = f.readline()
 | |
|         if not line: break
 | |
|         lineno = lineno + 1
 | |
|         while line[-2:] == '\\\n':
 | |
|             nextline = f.readline()
 | |
|             if not nextline: break
 | |
|             line = line + nextline
 | |
|             lineno = lineno + 1
 | |
|         newline = fixline(line)
 | |
|         if newline != line:
 | |
|             if g is None:
 | |
|                 try:
 | |
|                     g = open(tempname, 'w')
 | |
|                 except IOError as msg:
 | |
|                     f.close()
 | |
|                     err(tempname+': cannot create: '+
 | |
|                         str(msg)+'\n')
 | |
|                     return 1
 | |
|                 f.seek(0)
 | |
|                 lineno = 0
 | |
|                 initfixline()
 | |
|                 rep(filename + ':\n')
 | |
|                 continue # restart from the beginning
 | |
|             rep(repr(lineno) + '\n')
 | |
|             rep('< ' + line)
 | |
|             rep('> ' + newline)
 | |
|         if g is not None:
 | |
|             g.write(newline)
 | |
| 
 | |
|     # End of file
 | |
|     if filename == '-': return 0 # Done in filter mode
 | |
|     f.close()
 | |
|     if not g: return 0 # No changes
 | |
|     g.close()
 | |
| 
 | |
|     # Finishing touch -- move files
 | |
| 
 | |
|     # First copy the file's mode to the temp file
 | |
|     try:
 | |
|         statbuf = os.stat(filename)
 | |
|         os.chmod(tempname, statbuf[ST_MODE] & 0o7777)
 | |
|     except OSError as msg:
 | |
|         err(tempname + ': warning: chmod failed (' + str(msg) + ')\n')
 | |
|     # Then make a backup of the original file as filename~
 | |
|     try:
 | |
|         os.rename(filename, filename + '~')
 | |
|     except OSError as msg:
 | |
|         err(filename + ': warning: backup failed (' + str(msg) + ')\n')
 | |
|     # Now move the temp file to the original file
 | |
|     try:
 | |
|         os.rename(tempname, filename)
 | |
|     except OSError as msg:
 | |
|         err(filename + ': rename failed (' + str(msg) + ')\n')
 | |
|         return 1
 | |
|     # Return success
 | |
|     return 0
 | |
| 
 | |
| # Tokenizing ANSI C (partly)
 | |
| 
 | |
| Identifier = '(struct )?[a-zA-Z_][a-zA-Z0-9_]+'
 | |
| String = r'"([^\n\\"]|\\.)*"'
 | |
| Char = r"'([^\n\\']|\\.)*'"
 | |
| CommentStart = r'/\*'
 | |
| CommentEnd = r'\*/'
 | |
| 
 | |
| Hexnumber = '0[xX][0-9a-fA-F]*[uUlL]*'
 | |
| Octnumber = '0[0-7]*[uUlL]*'
 | |
| Decnumber = '[1-9][0-9]*[uUlL]*'
 | |
| Intnumber = Hexnumber + '|' + Octnumber + '|' + Decnumber
 | |
| Exponent = '[eE][-+]?[0-9]+'
 | |
| Pointfloat = r'([0-9]+\.[0-9]*|\.[0-9]+)(' + Exponent + r')?'
 | |
| Expfloat = '[0-9]+' + Exponent
 | |
| Floatnumber = Pointfloat + '|' + Expfloat
 | |
| Number = Floatnumber + '|' + Intnumber
 | |
| 
 | |
| # Anything else is an operator -- don't list this explicitly because of '/*'
 | |
| 
 | |
| OutsideComment = (Identifier, Number, String, Char, CommentStart)
 | |
| OutsideCommentPattern = '(' + '|'.join(OutsideComment) + ')'
 | |
| OutsideCommentProgram = re.compile(OutsideCommentPattern)
 | |
| 
 | |
| InsideComment = (Identifier, Number, CommentEnd)
 | |
| InsideCommentPattern = '(' + '|'.join(InsideComment) + ')'
 | |
| InsideCommentProgram = re.compile(InsideCommentPattern)
 | |
| 
 | |
| def initfixline():
 | |
|     global Program
 | |
|     Program = OutsideCommentProgram
 | |
| 
 | |
| def fixline(line):
 | |
|     global Program
 | |
| ##  print('-->', repr(line))
 | |
|     i = 0
 | |
|     while i < len(line):
 | |
|         match = Program.search(line, i)
 | |
|         if match is None: break
 | |
|         i = match.start()
 | |
|         found = match.group(0)
 | |
| ##      if Program is InsideCommentProgram: print(end='... ')
 | |
| ##      else: print(end='    ')
 | |
| ##      print(found)
 | |
|         if len(found) == 2:
 | |
|             if found == '/*':
 | |
|                 Program = InsideCommentProgram
 | |
|             elif found == '*/':
 | |
|                 Program = OutsideCommentProgram
 | |
|         n = len(found)
 | |
|         if found in Dict:
 | |
|             subst = Dict[found]
 | |
|             if Program is InsideCommentProgram:
 | |
|                 if not Docomments:
 | |
|                     print('Found in comment:', found)
 | |
|                     i = i + n
 | |
|                     continue
 | |
|                 if found in NotInComment:
 | |
| ##                  print(end='Ignored in comment: ')
 | |
| ##                  print(found, '-->', subst)
 | |
| ##                  print('Line:', line, end='')
 | |
|                     subst = found
 | |
| ##              else:
 | |
| ##                  print(end='Substituting in comment: ')
 | |
| ##                  print(found, '-->', subst)
 | |
| ##                  print('Line:', line, end='')
 | |
|             line = line[:i] + subst + line[i+n:]
 | |
|             n = len(subst)
 | |
|         i = i + n
 | |
|     return line
 | |
| 
 | |
| Docomments = 0
 | |
| def setdocomments():
 | |
|     global Docomments
 | |
|     Docomments = 1
 | |
| 
 | |
| Reverse = 0
 | |
| def setreverse():
 | |
|     global Reverse
 | |
|     Reverse = (not Reverse)
 | |
| 
 | |
| Dict = {}
 | |
| NotInComment = {}
 | |
| def addsubst(substfile):
 | |
|     try:
 | |
|         fp = open(substfile, 'r')
 | |
|     except IOError as msg:
 | |
|         err(substfile + ': cannot read substfile: ' + str(msg) + '\n')
 | |
|         sys.exit(1)
 | |
|     with fp:
 | |
|         lineno = 0
 | |
|         while 1:
 | |
|             line = fp.readline()
 | |
|             if not line: break
 | |
|             lineno = lineno + 1
 | |
|             try:
 | |
|                 i = line.index('#')
 | |
|             except ValueError:
 | |
|                 i = -1          # Happens to delete trailing \n
 | |
|             words = line[:i].split()
 | |
|             if not words: continue
 | |
|             if len(words) == 3 and words[0] == 'struct':
 | |
|                 words[:2] = [words[0] + ' ' + words[1]]
 | |
|             elif len(words) != 2:
 | |
|                 err(substfile + '%s:%r: warning: bad line: %r' % (substfile, lineno, line))
 | |
|                 continue
 | |
|             if Reverse:
 | |
|                 [value, key] = words
 | |
|             else:
 | |
|                 [key, value] = words
 | |
|             if value[0] == '*':
 | |
|                 value = value[1:]
 | |
|             if key[0] == '*':
 | |
|                 key = key[1:]
 | |
|                 NotInComment[key] = value
 | |
|             if key in Dict:
 | |
|                 err('%s:%r: warning: overriding: %r %r\n' % (substfile, lineno, key, value))
 | |
|                 err('%s:%r: warning: previous: %r\n' % (substfile, lineno, Dict[key]))
 | |
|             Dict[key] = value
 | |
| 
 | |
| if __name__ == '__main__':
 | |
|     main()
 | 
