X-Git-Url: http://git.megacz.com/?p=ghc-hetmet.git;a=blobdiff_plain;f=darcs-all;h=cdef628e19389783f66390a10a84734984a16f1d;hp=9efeab13e9d5f0cbcbd554eb4096c895d897f640;hb=0856ac59cfb455d32a3042317fdba0f5e85cab9c;hpb=cf570a334cdcaa03729c4a8b3e71c54eb37daf75 diff --git a/darcs-all b/darcs-all index 9efeab1..cdef628 100644 --- a/darcs-all +++ b/darcs-all @@ -1,94 +1,301 @@ -#!/bin/sh +#!/usr/bin/perl -w -set -e +use strict; -top_dirs="nofib testsuite" +# Usage: +# +# ./darcs-all [-q] [-s] [-i] [-r repo] +# [--nofib] [--testsuite] [--checked-out] cmd [darcs flags] +# +# Applies the darcs command "cmd" to each repository in the tree. +# +# e.g. +# ./darcs-all -r http://darcs.haskell.org/ghc get +# To get any repos which do not exist in the local tree +# +# ./darcs-all -r ~/ghc-validate push +# To push all your repos to the ~/ghc-validate tree +# +# ./darcs-all pull -a +# To pull everything from the default repos +# +# ./darc-all push --dry-run +# To see what local patches you have relative to the main repos +# +# -q says to be quite, and -s to be silent. +# +# -i says to ignore darcs errors and move on to the next repository +# +# -r repo says to use repo as the location of package repositories +# +# --checked-out says that the remote repo is in checked-out layout, as +# opposed to the layout used for the main repo. By default a repo on +# the local filesystem is assumed to be checked-out, and repos accessed +# via HTTP or SSH are assumed to be in the main repo layout; use +# --checked-out to override the latter. +# +# --nofib, --testsuite also get the nofib and testsuite repos respectively +# +# The darcs get flag you are most likely to want is --complete. By +# default we pass darcs the --partial flag. +# -default_repo_root="http://darcs.haskell.org/" -default_lib_repo_root=$default_repo_root/packages +$| = 1; # autoflush stdout after each print, to avoid output after die -quiet=NO +my $defaultrepo; -message() -{ - if [ "$quiet" = "NO" ]; then - echo $* - fi +my $verbose = 2; +my $ignore_failure = 0; +my $want_remote_repo = 0; +my $checked_out_flag = 0; + +my %tags; + +# Figure out where to get the other repositories from. +sub getrepo { + my $basedir = "."; + my $repo = $defaultrepo || `cat $basedir/_darcs/prefs/defaultrepo`; + chomp $repo; + + my $repo_base; + my $checked_out_tree; + + if ($repo =~ /^...*:/) { + # HTTP or SSH + # Above regex says "at least two chars before the :", to avoid + # catching Win32 drives ("C:\"). + $repo_base = $repo; + + # --checked-out is needed if you want to use a checked-out repo + # over SSH or HTTP + if ($checked_out_flag) { + $checked_out_tree = 1; + } else { + $checked_out_tree = 0; + } + + # Don't drop the last part of the path if specified with -r, as + # it expects repos of the form: + # + # http://darcs.haskell.org + # + # rather than + # + # http://darcs.haskell.org/ghc + # + if (!$defaultrepo) { + $repo_base =~ s#/[^/]+/?$##; + } + } + elsif ($repo =~ /^\/|\.\.\/|.:(\/|\\)/) { + # Local filesystem, either absolute or relative path + # (assumes a checked-out tree): + $repo_base = $repo; + $checked_out_tree = 1; + } + else { + die "Couldn't work out repo"; + } + + return $repo_base, $checked_out_tree; +} + +sub message { + if ($verbose >= 2) { + print "@_\n"; + } } -darcsall() -{ - message "== running darcs $* at the top level" - darcs $* - for dir in $top_dirs; do - if test -d $dir -a -d $dir/_darcs; then - message "== running darcs $* in $dir" - darcs $* --repodir $dir - else - message "== $dir not present or not a repository; skipping" - fi - done - for pkg in `cat libraries/core-packages libraries/extra-packages`; do - if test -d libraries/$pkg; then - message "== running darcs $* in libraries/$pkg" - darcs $* --repodir libraries/$pkg - else - echo "warning: $pkg doesn't seem to exist, use 'darcs-all get' to get it" - fi - done +sub warning { + if ($verbose >= 1) { + print "warning: @_\n"; + } } -darcsget() -{ - case $* in - *--partial*) ;; - *) echo "warning: adding --partial, to override use --complete" - esac - - repo_root=`cat _darcs/prefs/defaultrepo` - case $repo_root in - /*) lib_repos=$repo_root/libraries;; - *) lib_repos=$default_lib_repo_root;; - esac - - cd libraries - - if test "$extra" = "YES"; then - packages=`cat core-packages extra-packages` - else - packages=`cat core-packages` - fi - - for pkg in $packages; do - if test -d $pkg; then - echo "warning: $pkg already present; omitting" - else - repo=$lib_repos/$pkg - message "== running darcs get --partial $* $repo" - darcs get --partial $* $repo - fi - done +sub darcs { + message "== running darcs @_"; + system ("darcs", @_) == 0 + or $ignore_failure + or die "darcs failed: $?"; } -if test ! -d _darcs -o ! -d compiler; then - echo "error: darcs-all must be run from the top level of the ghc tree." - exit 1; -fi - -case $* in - *-q*) quiet=YES;; -esac - -# --extra says we grab the extra libs with 'get'. It has no effect on -# the other commands. -extra=NO; -case $1 in - --extra) shift; extra=YES; -esac - -case $1 in - get) shift; darcsget $*;; - # Hack around whatsnew failing if there are no changes - w|wh|wha|what|whats|whatsn|whatsne|whatsnew) set +e; darcsall $*;; - *) darcsall $*;; -esac +sub darcsall { + my $localpath; + my $remotepath; + my $path; + my $tag; + my @repos; + + my ($repo_base, $checked_out_tree) = getrepo(); + + open IN, "< packages" or die "Can't open packages file"; + @repos = ; + close IN; + + REPO: foreach (@repos) { + chomp; + if (/^([^# ]+) +(?:([^ ]+) +)?([^ ]+) +([^ ]+)$/) { + $localpath = $1; + $tag = defined($2) ? $2 : ""; + $remotepath = $3; + + if ($checked_out_tree) { + $path = "$repo_base/$localpath"; + } + else { + $path = "$repo_base/$remotepath"; + } + + if (-d "$localpath/_darcs") { + if ($want_remote_repo) { + darcs (@_, "--repodir", $localpath, $path); + } else { + darcs (@_, "--repodir", $localpath); + } + } + elsif ($tag eq "") { + message "== Required repo $localpath is missing! Skipping"; + } + else { + message "== $localpath repo not present; skipping"; + } + } + elsif (! /^(#.*)?$/) { + die "Bad line: $_"; + } + } +} + +sub darcsget { + my $r_flags; + my $localpath; + my $remotepath; + my $path; + my $tag; + my @repos; + + my ($repo_base, $checked_out_tree) = getrepo(); + + if (! grep /(?:--complete|--partial)/, @_) { + warning("adding --partial, to override use --complete"); + $r_flags = [@_, "--partial"]; + } + else { + $r_flags = \@_; + } + + open IN, "< packages" or die "Can't open packages file"; + @repos = ; + close IN; + + foreach (@repos) { + chomp; + if (/^([^ ]+) +(?:([^ ]+) +)?([^ ]+) +([^ ]+)$/) { + $localpath = $1; + $tag = defined($2) ? $2 : ""; + $remotepath = $3; + + if ($checked_out_tree) { + $path = "$repo_base/$localpath"; + } + else { + $path = "$repo_base/$remotepath"; + } + + if (($tag eq "") || defined($tags{$tag})) { + if (-d $localpath) { + warning("$localpath already present; omitting"); + } + else { + darcs (@$r_flags, $path, $localpath); + } + } + } + elsif (! /^(#.*)?$/) { + die "Bad line: $_"; + } + } +} + +sub main { + if (! -d "compiler") { + die "error: darcs-all must be run from the top level of the ghc tree." + } + + while ($#_ ne -1) { + my $arg = shift; + # We handle -q here as well as lower down as we need to skip over it + # if it comes before the darcs command + if ($arg eq "-q") { + $verbose = 1; + } + elsif ($arg eq "-s") { + $verbose = 0; + } + elsif ($arg eq "-r") { + $defaultrepo = shift; + } + elsif ($arg eq "-i") { + $ignore_failure = 1; + } + # --nofib tells get to also grab the nofib repo. + # It has no effect on the other commands. + elsif ($arg eq "--nofib") { + $tags{"nofib"} = 1; + } + # --testsuite tells get to also grab the testsuite repo. + # It has no effect on the other commands. + elsif ($arg eq "--testsuite") { + $tags{"testsuite"} = 1; + } + elsif ($arg eq "--checked-out") { + $checked_out_flag = 1; + } + else { + unshift @_, $arg; + if (grep /^-q$/, @_) { + $verbose = 1; + } + last; + } + } + + if ($#_ eq -1) { + die "What do you want to do?"; + } + my $command = $_[0]; + if ($command eq "get") { + darcsget @_; + } + else { + if ($command =~ /^(?:w|wh|wha|what|whats|whatsn|whatsne|whatsnew)$/) { + # Hack around whatsnew failing if there are no changes + $ignore_failure = 1; + } + if ($command =~ /^(pul|pus|sen|put)/) { + $want_remote_repo = 1; + } + darcsall @_; + } +} + +END { + message "== Checking for old bytestring repo"; + if (-d "libraries/bytestring/_darcs") { + if ((system "darcs annotate --repodir libraries/bytestring --match 'hash 20080118173113-3fd76-d5b74c04372a297b585ebea4e16d524551ce5035' > /dev/null 2> /dev/null") == 0) { + print <