Commit 704be878 authored by Wei-Yin Chen (陳威尹)'s avatar Wei-Yin Chen (陳威尹) Committed by Commit Bot

Add reflink support for gclient-new-workdir.py

On filesystems supporting copy-on-write, like Btrfs and ZFS, use
"cp --reflink" to achieve faster speed and lower disk space usage.

Option --reflink and --no-reflink can override the automatic detection.

When reflink is supported, toolchains, support libraries, and artifacts
would also be preserved. This means you can do incremental builds right
after creating a new workdir.

For Linux Chromium checkout, one additional workdir only takes 0.8~1 GB
disk space in Btrfs, including data and metadata, depending on how many
artifacts are there inside the out*/ directories.

Change-Id: I8b913efba9e5afaeea2adc1c10a561f63cb50282
Bug: 721585
Reviewed-on: https://chromium-review.googlesource.com/499567
Commit-Queue: Wei-Yin Chen (陳威尹) <wychen@chromium.org>
Reviewed-by: 's avatarRobbie Iannucci <iannucci@chromium.org>
parent 05203496
...@@ -27,6 +27,12 @@ def parse_options(): ...@@ -27,6 +27,12 @@ def parse_options():
parser.add_argument('repository', type=os.path.abspath, parser.add_argument('repository', type=os.path.abspath,
help='should contain a .gclient file') help='should contain a .gclient file')
parser.add_argument('new_workdir', help='must not exist') parser.add_argument('new_workdir', help='must not exist')
parser.add_argument('--reflink', action='store_true', default=None,
help='''force to use "cp --reflink" for speed and disk
space. need supported FS like btrfs or ZFS.''')
parser.add_argument('--no-reflink', action='store_false', dest='reflink',
help='''force not to use "cp --reflink" even on supported
FS like btrfs or ZFS.''')
args = parser.parse_args() args = parser.parse_args()
if not os.path.exists(args.repository): if not os.path.exists(args.repository):
...@@ -42,21 +48,52 @@ def parse_options(): ...@@ -42,21 +48,52 @@ def parse_options():
return args return args
def support_cow(src, dest):
try:
subprocess.check_output(['cp', '-a', '--reflink', src, dest],
stderr=subprocess.STDOUT)
except subprocess.CalledProcessError:
return False
finally:
os.remove(dest)
return True
def main(): def main():
args = parse_options() args = parse_options()
gclient = os.path.join(args.repository, '.gclient') gclient = os.path.join(args.repository, '.gclient')
new_gclient = os.path.join(args.new_workdir, '.gclient')
os.makedirs(args.new_workdir) os.makedirs(args.new_workdir)
os.symlink(gclient, os.path.join(args.new_workdir, '.gclient')) if args.reflink is None:
args.reflink = support_cow(gclient, new_gclient)
if args.reflink:
print('Copy-on-write is supported. Using reflink to copy the repo.')
os.symlink(gclient, new_gclient)
for root, dirs, _ in os.walk(args.repository): for root, dirs, _ in os.walk(args.repository):
if '.git' in dirs: if '.git' in dirs:
workdir = root.replace(args.repository, args.new_workdir, 1) workdir = root.replace(args.repository, args.new_workdir, 1)
print('Creating: %s' % workdir) print('Creating: %s' % workdir)
if args.reflink:
if not os.path.exists(workdir):
print('Copying: %s' % workdir)
subprocess.check_call(['cp', '-a', '--reflink', root, workdir])
shutil.rmtree(os.path.join(workdir, '.git'))
git_common.make_workdir(os.path.join(root, '.git'), git_common.make_workdir(os.path.join(root, '.git'),
os.path.join(workdir, '.git')) os.path.join(workdir, '.git'))
subprocess.check_call(['git', 'checkout', '-f'], cwd=workdir) if args.reflink:
subprocess.check_call(['cp', '-a', '--reflink',
os.path.join(root, '.git', 'index'),
os.path.join(workdir, '.git', 'index')])
else:
subprocess.check_call(['git', 'checkout', '-f'], cwd=workdir)
if args.reflink:
subprocess.check_call(['git', 'clean', '-df'], cwd=workdir)
if __name__ == '__main__': if __name__ == '__main__':
......
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment