#!/usr/bin/python
#
#===- clang-format-diff.py - ClangFormat Diff Reformatter ----*- python -*--===#
#
# The LLVM Compiler Infrastructure
#
# This file is distributed under the University of Illinois Open Source
# License. See LICENSE.TXT for details.
#
#===------------------------------------------------------------------------===#
r"""
ClangFormat Diff Reformatter
============================
This script reads input from a unified diff and reformats all the changed
lines. This is useful to reformat all the lines touched by a specific patch.
Example usage for git users:
git diff -U0 HEAD^ | clang-format-diff.py -p1
"""
import argparse
import re
import subprocess
import sys
# Change this to the full path if clang-format is not on the path.
binary = 'clang-format'
def getOffsetLength(filename, line_number, line_count):
"""
Calculates the field offset and length based on line number and count.
"""
offset = 0
length = 0
with open(filename, 'r') as f:
for line in f:
if line_number > 1:
offset += len(line)
line_number -= 1
elif line_count > 0:
length += len(line)
line_count -= 1
else:
break
return offset, length
def formatRange(r, style):
"""
Formats range 'r' according to style 'style'.
"""
filename, line_number, line_count = r
# FIXME: Add other types containing C++/ObjC code.
if not (filename.endswith(".cpp") or filename.endswith(".cc") or
filename.endswith(".h")):
return
offset, length = getOffsetLength(filename, line_number, line_count)
with open(filename, 'r') as f:
text = f.read()
command = [binary, '-offset', str(offset), '-length', str(length)]
if style:
command.extend(['-style', style])
p = subprocess.Popen(command, stdout=subprocess.PIPE, stderr=subprocess.PIPE,
stdin=subprocess.PIPE)
stdout, stderr = p.communicate(input=text)
if stderr:
print stderr
return
if not stdout:
print 'Segfault occurred while formatting', filename
print 'Please report a bug on llvm.org/bugs.'
return
with open(filename, 'w') as f:
f.write(stdout)
def main():
parser = argparse.ArgumentParser(description=
'Reformat changed lines in diff')
parser.add_argument('-p', default=0,
help='strip the smallest prefix containing P slashes')
parser.add_argument('-style',
help='formatting style to apply (LLVM, Google, Chromium)')
args = parser.parse_args()
filename = None
ranges = []
for line in sys.stdin:
match = re.search('^\+\+\+\ (.*?/){%s}(\S*)' % args.p, line)
if match:
filename = match.group(2)
if filename == None:
continue
match = re.search('^@@.*\+(\d+)(,(\d+))?', line)
if match:
line_count = 1
if match.group(3):
line_count = int(match.group(3))
ranges.append((filename, int(match.group(1)), line_count))
# Reverse the ranges so that the reformatting does not influence file offsets.
for r in reversed(ranges):
# Do the actual formatting.
formatRange(r, args.style)
if __name__ == '__main__':
main()