]> git.pld-linux.org Git - packages/rpm-build-tools.git/blobdiff - adapter.awk
- update
[packages/rpm-build-tools.git] / adapter.awk
index e4dab2bb13a49db7311478a8ac96db7d0f433222..fd5cd795d58d7945ffb3de339c0e2ca91cd30260 100644 (file)
@@ -1,98 +1,81 @@
-#!/bin/awk -f
+#!/usr/bin/gawk -f
 #
-# This is adapter v0.27. Adapter adapts .spec files for PLD Linux.
-# $Id$
+# Adapter adapts .spec files for PLD Linux.
 #
-# Copyright (C) 1999-2006 PLD-Team <feedback@pld-linux.org>
+# Copyright (C) 1999-2011 PLD-Team <feedback@pld-linux.org>
 # Authors:
-#      Micha³ Kuratczyk <kura@pld.org.pl>
+#      Michał Kuratczyk <kura@pld.org.pl>
 #      Sebastian Zagrodzki <s.zagrodzki@mimuw.edu.pl>
-#      Tomasz K³oczko <kloczek@rudy.mif.pg.gda.pl>
+#      Tomasz Kłoczko <kloczek@rudy.mif.pg.gda.pl>
 #      Artur Frysiak <wiget@pld-linux.org>
 #      Michal Kochanowicz <mkochano@pld.org.pl>
-#      Elan Ruusamä¤e <glen@pld-linux.org>
+#      Jakub Bogusz <qboosh@pld-linux.org>
+#      Elan Ruusamäe <glen@pld-linux.org>
 #
 # See cvs log adapter{,.awk} for list of contributors
 #
 # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # #
 
 # TODO
+# - really long sourceX make preamble sorting totally fcked up (try snake.spec r1.1)
 # - parse ../PLD-doc/BuildRequires.txt and setup proper BR epoches?
 # - add "-nc" option to skip CVS interaction
 # - sort Summary(XX)
 # - sort Requires, BuildRequires
 # - check if %description (lang=C) contains 8bit
 # - desc wrapping is totally fucked up on global.spec,1.25, dosemu.spec,1.115-
+# - it should change: /%source([0-9]+)/i to %{SOURCE\1}
+# - extra quote on LDFLAGS line: https://bugs.launchpad.net/pld-linux/+bug/385836
+# - %{with_foo:%attr()...} gets converted to %attr() %{with_foo:...} [vlc.spec]
+# - 'R: foo ' (with traliling space) gets coverted to "R: foo\nR: " [vlc.spec @ 1.199 ]
 
 BEGIN {
        RPM_SECTIONS = "package|build|changelog|clean|description|install|post|posttrans|postun|pre|prep|pretrans|preun|triggerin|triggerpostun|triggerun|verifyscript|check"
        SECTIONS = "^%(" RPM_SECTIONS ")"
 
-       PREAMBLE_TAGS = "(Summary|Name|Version|Release|Epoch|License|Group|URL|BuildArch|BuildRoot|Obsoletes|Conflicts|Provides|ExclusiveArch|ExcludeArch|Pre[Rr]eq|(Build)?Requires)"
+       RCSID = "$Id$"
+       rev = RCSID # TODO: parse from RCSID
+       VERSION = "0.35/" rev
 
-       preamble = 1            # Is it part of preamble? Default - yes
+       PREAMBLE_TAGS = "(R|BR|Summary|Name|Version|Release|Epoch|License|Group|URL|BuildArch|BuildRoot|Obsoletes|Conflicts|Provides|ExclusiveArch|ExcludeArch|Pre[Rr]eq|(Build)?Requires|Suggests|Auto(Req|Prov))"
+
+       usedigest = 0   # Enable to switch to rpm 4.4.6+ md5 digests
+
+       preamble = 1    # Is it part of preamble? Default - yes
        boc = 4                 # Beginning of %changelog
        bod = 0                 # Beginning of %description
        tw = 70                 # Descriptions width
 
        b_idx = 0               # index of BR/R arrays
+       BR_count = 0    # number of additional BuildRequires
 
        # If variable removed, then 1 (for removing it from export)
        removed["LDFLAGS"] = 0
        removed["CFLAGS"] = 0
        removed["CXXFLAGS"] = 0
 
-       # get cvsaddress for changelog section
-       # using rpm macros as too lazy to add ~/.adapterrc parsing support.
-       "rpm --eval '%{?_cvsmaildomain}%{!?_cvsmaildomain:@pld-linux.org}'" | getline _cvsmaildomain
-       "rpm --eval '%{?_cvsmailfeedback}%{!?_cvsmailfeedback:PLD Team <feedback@pld-linux.org>}'" | getline _cvsmailfeedback
-
        # If 1, we are inside of comment block (started with /^#%/)
        comment_block = 0
 
-       # File with rpm groups
-       "rpm --eval %_sourcedir" | getline groups_file
-       groups_file = groups_file "/rpm.groups"
-       system("cd `rpm --eval %_sourcedir`; [ -f rpm.groups ] || cvs up rpm.groups >/dev/null")
+       import_rpm_macros()
+
+       packages_dir = topdir
+       groups_file = packages_dir "/rpm.groups"
+
+       system("cd "packages_dir"; [ -f rpm.groups ] || cvs up rpm.groups > /dev/null")
        system("[ -d ../PLD-doc ] && cd ../PLD-doc && ([ -f BuildRequires.txt ] || cvs up BuildRequires.txt >/dev/null)");
 
        # Temporary file for changelog section
-       changelog_file = ENVIRON["HOME"] "/tmp/adapter.changelog"
-
-       # Load rpm macros
-       "rpm --eval %_prefix"   | getline prefix
-       "rpm --eval %_bindir"   | getline bindir
-       "rpm --eval %_sbindir"  | getline sbindir
-       "rpm --eval %_libdir"   | getline libdir
-       "rpm --eval %_sysconfdir" | getline sysconfdir
-       "rpm --eval %_datadir"  | getline datadir
-       "rpm --eval %_includedir" | getline includedir
-       "rpm --eval %_mandir"   | getline mandir
-       "rpm --eval %_infodir"  | getline infodir
-       "rpm --eval %_examplesdir"      | getline examplesdir
-       "rpm --eval %_defaultdocdir"    | getline docdir
-       "rpm --eval %_desktopdir" | getline desktopdir
-       "rpm --eval %_pixmapsdir" | getline pixmapsdir
-
-       "rpm --eval %perl_sitearch" | getline perl_sitearch
-       "rpm --eval %perl_archlib" | getline perl_archlib
-       "rpm --eval %perl_privlib" | getline perl_privlib
-       "rpm --eval %perl_vendorlib" | getline perl_vendorlib
-       "rpm --eval %perl_vendorarch" | getline perl_vendorarch
-       "rpm --eval %perl_sitelib" | getline perl_sitelib
-
-       "rpm --eval %py_sitescriptdir" | getline py_sitescriptdir
-       "rpm --eval %py_sitedir" | getline py_sitedir
-       "rpm --eval %py_scriptdir " | getline py_scriptdir
-       "rpm --eval %php_pear_dir" | getline php_pear_dir
-       "rpm --eval %tmpdir" | getline tmpdir
+       changelog_file = mktemp("adapter.changelogXXXXXX")
 }
 
 # There should be a comment with CVS keywords on the first line of file.
 FNR == 1 {
-       if (!/# \$Revision:/)   # If this line is already OK?
+       if (!/^# \$[R]evision: .* \$, \$[D]ate: .* \$$/) {      # If this line is already OK?
                print "# $" "Revision:$, " "$" "Date:$" # No
-       else {
+               if ( /^#.*([rR]evision|[dD]ate)/ ) # There was something similar, but incorrect
+                        next
+       } else {
                print $0                                # Yes
                next            # It is enough for first line
        }
@@ -110,34 +93,46 @@ defattr == 1 {
        defattr = 0
 }
 
+# call mktemp(1) and return the value
+function mktemp(template, tmp) {
+   "mktemp " template | getline tmp
+   return tmp
+}
+
 function b_makekey(a, b,       s) {
        s = a "" b;
        # kill bcond
-    gsub(/[#%]+{[!?]+[_a-zA-Z0-9]+:/, "", s);
+       gsub(/[#%]+{[!?]+[_a-zA-Z0-9]+:/, "", s);
 
        # kill commented out items
-    gsub(/^#[ \t]*/, "", s);
+       gsub(/^#[ \t]*/, "", s);
 
        # force order
-    gsub(/^Summary\(/, "11Summary(", s);
-    gsub(/^Summary/, "10Summary", s);
-    gsub(/^Name/, "2Name", s);
-    gsub(/^Version/, "3Version", s);
-    gsub(/^Release/, "4Release", s);
-    gsub(/^Epoch/, "5Epoch", s);
-    gsub(/^License/, "5License", s);
-    gsub(/^Group/, "6Group", s);
-    gsub(/^URL/, "7URL", s);
-
-    gsub(/^BuildRequires/, "B1BuildRequires", s);
-    gsub(/^BuildConflicts/, "B2BuildConflicts", s);
-    gsub(/^Provides/, "X1Provides", s);
-    gsub(/^Obsoletes/, "X2Obsoletes", s);
-    gsub(/^Conflicts/, "X3Conflicts", s);
-    gsub(/^BuildArch/, "X4BuildArch", s);
-    gsub(/^ExclusiveArch/, "X6ExclusiveArch", s);
-    gsub(/^ExcludeArch/, "X7ExcludeArch", s);
-    gsub(/^BuildRoot/, "X9BuildRoot", s);
+       gsub(/^Summary\(/, "11Summary(", s);
+       gsub(/^Summary/, "10Summary", s);
+
+       gsub(/^Name/, "2Name", s);
+       gsub(/^Version/, "3Version", s);
+       gsub(/^Release/, "4Release", s);
+       gsub(/^Epoch/, "5Epoch", s);
+       gsub(/^License/, "5License", s);
+       gsub(/^Group/, "6Group", s);
+       gsub(/^URL/, "7URL", s);
+
+       gsub(/^BuildRequires/, "B1BuildRequires", s);
+       gsub(/^BuildConflicts/, "B2BuildConflicts", s);
+
+       gsub(/^Suggests/, "X1Suggests", s);
+       gsub(/^Provides/, "X2Provides", s);
+       gsub(/^Obsoletes/, "X3Obsoletes", s);
+       gsub(/^Conflicts/, "X4Conflicts", s);
+       gsub(/^BuildArch/, "X5BuildArch", s);
+       gsub(/^ExclusiveArch/, "X6ExclusiveArch", s);
+       gsub(/^ExcludeArch/, "X7ExcludeArch", s);
+       gsub(/^BuildRoot/, "X9BuildRoot", s);
+
+       gsub(/^AutoProv/, "Xx1AutoProv", s);
+       gsub(/^AutoReq/, "Xx2AutoReq", s);
 
 #      printf("%s -> %s\n", a""b, s);
        return s;
@@ -150,13 +145,17 @@ function b_makekey(a, b,  s) {
                while (getline)         # print the rest of spec as it is
                        print
                do_not_touch_anything = 1 # do not touch anything in END()
-               exit 0
+               exit(rc = 0)
        }
 
        # Generally, comments are printed without touching
        sub(/[ \t]+$/, "")
 
-       if (/Source.*md5/) {
+       if (/#[ \t]*Source.*md5/) {
+               if (usedigest == 1) {
+                       sub(/^#[ \t]*Source/, "BuildRequires:\tdigest(%SOURCE", $0)
+                       sub(/-md5[ \t]*:[ \t]*/, ") = ", $0)
+               }
                print $0
                next
        }
@@ -232,6 +231,8 @@ function b_makekey(a, b,    s) {
                _pre = $3
        if ($2 ~ /^_snap$/)
                _snap = $3
+       if ($2 ~ /^subver$/)
+               subver = $3
 
        # these are used usually when adapterizing external spec
        if ($2 ~ /^name$/)
@@ -243,6 +244,22 @@ function b_makekey(a, b,   s) {
 
        if ($2 ~ /^mod_name$/)
                mod_name = $3
+       if ($2 ~ /^_?pearname$/)
+               pearname = $3
+       if ($2 ~ /^_class$/)
+               pear_class = $3
+       if ($2 ~ /^_subclass$/)
+               pear_subclass = $3
+
+       # kill the _class and _subclass pear macros
+       if ($2 == "_pearname" || $2 == "pearname") {
+               if (pear_class) {
+                       gsub("%{_class}", pear_class, $3);
+               }
+               if (pear_subclass) {
+                       gsub("%{_subclass}", pear_subclass, $3);
+               }
+       }
 
        sub(/[ \t]+$/, "");
        # do nothing further, otherwise adapter thinks we're at preamble
@@ -318,7 +335,7 @@ function b_makekey(a, b,    s) {
                sub(/^%setup/, "%setup -q")
        }
 
-       if (/^%setup/) {
+       if (/^%setup/ && name != "setup") {
                $0 = fixedsub(name, "%{name}", $0);
                $0 = fixedsub(version, "%{version}", $0);
                if (_beta) {
@@ -333,11 +350,20 @@ function b_makekey(a, b,  s) {
                if (_snap) {
                        $0 = fixedsub(_snap, "%{_snap}", $0);
                }
+               if (subver) {
+                       $0 = fixedsub(subver, "%{subver}", $0);
+               }
        }
 
        if (/^%setup/ && /-n %{name}-%{version}( |$)/) {
                $0 = fixedsub(" -n %{name}-%{version}", "", $0)
        }
+       sub("^%patch ", "%patch0 ");
+
+       # fedora extras
+       if (/^%apply/) {
+               sub("^%apply -n", "%patch");
+       }
 
        # invalid in %prep
        sub("^rm -rf \$RPM_BUILD_ROOT.*", "");
@@ -408,6 +434,7 @@ function b_makekey(a, b,    s) {
        $0 = fixedsub("automake --add-missing --copy", "%{__automake}", $0);
        $0 = fixedsub("automake -a --foreign --copy", "%{__automake}", $0);
        $0 = fixedsub("automake -a -c --foreign", "%{__automake}", $0);
+       $0 = fixedsub("automake -a -c", "%{__automake}", $0);
        $0 = fixedsub("libtoolize --force --automake --copy", "%{__libtoolize}", $0);
        $0 = fixedsub("libtoolize -c -f --automake", "%{__libtoolize}", $0);
 
@@ -415,10 +442,14 @@ function b_makekey(a, b,  s) {
        sub(/^autoheader$/, "%{__autoheader}");
        sub(/^autoconf$/, "%{__autoconf}");
        sub(/^automake$/, "%{__automake}");
+       sub(/^libtoolize$/, "%{__libtoolize}");
 
        # atrpms
        $0 = fixedsub("%perl_configure", "%{__perl} Makefile.PL \\\n\tINSTALLDIRS=vendor", $0);
        $0 = fixedsub("%perl_makecheck", "%{?with_tests:%{__make} test}", $0);
+
+       # alt linux
+       $0 = fixedsub("%make_build", "%{__make}", $0);
 }
 
 ##########
@@ -427,10 +458,7 @@ function b_makekey(a, b,   s) {
 /^%clean/, (!/^%clean/ && $0 ~ SECTIONS) {
        did_clean = 1
 
-       # prevent next section header like "%post -p /sbin/ldconfig" being adapterized
-       if (!/^%post/) {
-               use_macros()
-       }
+       use_macros()
 }
 
 ############
@@ -442,8 +470,8 @@ function b_makekey(a, b,    s) {
 
        # foreign rpms
        sub("^%{__rm} -rf %{buildroot}", "rm -rf $RPM_BUILD_ROOT")
-    sub("%buildroot", "$RPM_BUILD_ROOT");
-    sub("%{buildroot}", "$RPM_BUILD_ROOT");
+       sub("%buildroot", "$RPM_BUILD_ROOT");
+       sub("%{buildroot}", "$RPM_BUILD_ROOT");
 
        if (/^[ \t]*rm([ \t]+-[rf]+)*[ \t]+(\${?RPM_BUILD_ROOT}?|%{?buildroot}?)/ && did_rmroot==0) {
                did_rmroot=1
@@ -456,10 +484,14 @@ function b_makekey(a, b,  s) {
                did_rmroot=1
        }
 
-    if (tmpdir) {
-        buildroot = tmpdir "/" name "-" version "-root-" ENVIRON["USER"]
+       if (tmpdir) {
+               buildroot = tmpdir "/" name "-" version "-root-" ENVIRON["USER"]
                gsub(buildroot, "$RPM_BUILD_ROOT")
-    }
+       }
+
+       if (!/%{_lib}/) {
+               sub("\$RPM_BUILD_ROOT/%", "$RPM_BUILD_ROOT%")
+       }
 
        use_macros()
 
@@ -467,9 +499,8 @@ function b_makekey(a, b,    s) {
        if (/mkdir -p/)
                sub(/mkdir -p/, "install -d")
 
-       # 'install' instead 'cp -p'
-       if (/cp -p\b/)
-               sub(/cp -p/, "install")
+       # cp -a already implies cp -r
+       sub(/^cp -ar/, "cp -a")
 
        # No '-u root' or '-g root' for 'install'
        if (/^install/ && /-[ug][ \t]*root/)
@@ -488,6 +519,9 @@ function b_makekey(a, b,    s) {
 
        # atrpms
        $0 = fixedsub("%perl_makeinstall", "%{__make} pure_install \\\n\tDESTDIR=$RPM_BUILD_ROOT", $0);
+
+       # alt linux
+       $0 = fixedsub("%make_install DESTDIR=$RPM_BUILD_ROOT install", "%{__make} install \\\n\tDESTDIR=$RPM_BUILD_ROOT", $0);
 }
 
 ##########
@@ -500,7 +534,9 @@ function b_makekey(a, b,    s) {
        if ($0 ~ /^%files/)
                defattr = 1
 
-       use_files_macros()
+       if (!use_files_macros()) {
+               next
+       }
 }
 
 ##############
@@ -512,24 +548,27 @@ function b_makekey(a, b,  s) {
        skip = 0
        # There should be some CVS keywords on the first line of %changelog.
        if (boc == 3) {
-               if ($0 !~ _cvsmailfeedback)
+               if ($0 !~ _cvsmailfeedback) {
                        print "* %{date} " _cvsmailfeedback > changelog_file
-               else
+               } else {
                        skip = 1
+               }
                boc = 2
        }
        if (boc == 2 && !skip) {
                if (!/All persons listed below/) {
                        printf "All persons listed below can be reached at " > changelog_file
                        print "<cvs_login>" _cvsmaildomain "\n" > changelog_file
-               } else
+               } else {
                        skip = 1
+               }
                boc = 1
        }
        if (boc == 1 && !skip) {
                if (!/^$/) {
-                       if (!/\$.*Log:.*\$/)
+                       if (!/\$.*Log:.*\$/) {
                                print "$" "Log:$" > changelog_file
+                       }
                        boc = 0
                }
        }
@@ -543,11 +582,19 @@ function b_makekey(a, b,  s) {
                boc = 3
        }
 
-       sub(/[ \t]+$/, "")
-       if (!/^%[a-z]+$/ || /changelog/)
+       sub(/[ \t]+$/, "");
+       if (!/^%[a-z]+$/ || /changelog/) {
+               # stop changelog if "real" changelog starts
+               if (boc == 0 && /^\* /) {
+                       boc = -1
+               }
+               if (boc == -1) {
+                       next;
+               }
                print > changelog_file
-       else
+       } else {
                print
+       }
        next
 }
 
@@ -562,6 +609,19 @@ function b_makekey(a, b,   s) {
                sub(" >/dev/null 2>&1 \|\|:", "");
        }
 
+       # fedora extras macros
+       if (/%__fe_useradd/) {
+               sub("%__fe_useradd", "%useradd -u ");
+               sub(" 2> /dev/null \|\| :", "");
+               sub(" >/dev/null 2>&1 \|\|:", "");
+               sub(" &>/dev/null \\|\\| :", "");
+       }
+
+       if (/%__fe_groupadd/) {
+               sub("%__fe_groupadd", "%groupadd -g ");
+               sub(" &>/dev/null \\|\\| :", "");
+       }
+
        # %useradd and %groupadd may not be wrapped
        if (/%(useradd|groupadd).*\\$/) {
                a = $0; getline;
@@ -573,14 +633,36 @@ function b_makekey(a, b,  s) {
 
 /^%post/, (!/^%post/ && $0 ~ SECTIONS) {
        preamble = 0
+
+       # fedora extras macros
+       sub("%__chkconfig", "/sbin/chkconfig");
+
+       sub("update-desktop-database &> /dev/null \\|\\| :", "%update_desktop_database");
+       sub("touch --no-create %{_datadir}/icons/hicolor", "%update_icon_cache_post hicolor");
+       sub("if \\[ -x %{_bindir}/gtk-update-icon-cache \\]; then\n\t%{_bindir}/gtk-update-icon-cache -q %{_datadir}/icons/hicolor \|\| :\nfi", "");
+
+       sub("export GCONF_CONFIG_SOURCE=`gconftool-2 --get-default-source`", "")
+       if (/gconftool-2 --makefile-install-rule/) {
+               sub("gconftool-2 --makefile-install-rule %{_sysconfdir}/gconf/schemas/", "%gconf_schema_install ")
+               sub("> /dev/null", "");
+       }
+
        use_macros()
 }
 /^%preun/, (!/^%preun/ && $0 ~ SECTIONS) {
        preamble = 0
-       use_script_macros()
+       use_macros()
 }
 /^%postun/, (!/^%postun/ && $0 ~ SECTIONS) {
        preamble = 0
+
+       # fedora extras macros
+       if (/%__fe_userdel|%__fe_groupdel/) {
+               sub("%__fe_groupdel", "%groupremove");
+               sub("%__fe_userdel", "%userremove");
+               sub(" &>/dev/null \\|\\| :", "");
+       }
+
        use_script_macros()
 }
 /^%triggerin/, (!/^%triggerin/ && $0 ~ SECTIONS) {
@@ -627,8 +709,7 @@ preamble == 1 {
        }
 
        field = tolower($1)
-       fieldnlower = $1
-       if (field ~ /summary:/ && !/etc\.$/) {
+       if (field ~ /summary:/ && !/etc\.$/ && !/Inc\.$/) {
                sub(/\.$/, "", $0);
        }
        if (field ~ /group(\([^)]+\)):/)
@@ -637,38 +718,7 @@ preamble == 1 {
                format_preamble()
                group = $0;
                sub(/^[^ \t]*[ \t]*/, "", group);
-
-               sub(/^Utilities\//,"Applications/", group)
-               sub(/^Games/,"Applications/Games", group)
-               sub(/^X11\/Games/,"X11/Applications/Games", group)
-               sub(/^X11\/GNOME\/Development\/Libraries/,"X11/Development/Libraries", group)
-               sub(/^X11\/GNOME\/Applications/,"X11/Applications", group)
-               sub(/^X11\/GNOME/,"X11/Applications", group)
-               sub(/^X11\/Utilities/,"X11/Applications", group)
-               sub(/^X11\/Games\/Strategy/,"X11/Applications/Games/Strategy", group)
-               sub(/^Shells/,"Applications/Shells", group)
-               sub(/^System Environment\/Libraries$/, "Libraries", group)
-               sub(/^System Environment\/Daemons$/, "Daemons", group)
-               sub(/^Applications\/Internet$/, "Applications/Networking", group)
-               sub(/^Applications\/Daemons$/, "Daemons", group)
-               sub(/^Application\/Multimedia$/, "Applications/Multimedia", group)
-               sub(/^System\/Servers$/, "Daemons", group)
-               sub(/^X11\/Xserver$/, "X11/Servers", group)
-               sub(/^X11\/XFree86/, "X11", group)
-               sub(/^Applications\/Compilers$/, "Development/Languages", group)
-               sub(/^Applications\/Internet\/Peer to Peer/, "Applications/Networking", group)
-               sub(/^Networking\/Deamons$/, "Networking/Daemons", group)
-               sub(/^Development\/Docs$/, "Documentation", group)
-               sub(/^Development\/Documentation$/, "Documentation", group)
-               sub(/^System Environment\/Kernel$/, "Base/Kernel", group)
-               sub(/^Development\/Libraries\/Java$/, "Development/Languages/Java", group)
-               sub(/^Development\/Java/, "Development/Languages/Java", group)
-               sub(/^Development\/Testing$/, "Development", group)
-               sub(/^Text Processing\/Markup\/HTML$/, "Applications/Text", group)
-               sub(/^Text Processing\/Markup\/XML$/, "Applications/Text", group)
-               sub(/^Web\/Database$/, "Applications/WWW", group)
-               sub(/^System Environment\/Base$/, "Base", group)
-
+               group = replace_groupnames(group);
                $0 = "Group:\t\t" group
 
                if (group ~ /^X11/ && x11 == 0) # Is it X11 application?
@@ -698,7 +748,7 @@ preamble == 1 {
        }
 
        # split (build)requires, obsoletes on commas
-       if (field ~ /(obsoletes|requires|provides|conflicts):/ && NF > 2) {
+       if (field ~ /(obsoletes|requires|provides|conflicts|suggests):/ && NF > 2) {
                value = substr($0, index($0, $2));
                $0 = format_requires($1, value);
        }
@@ -726,18 +776,14 @@ preamble == 1 {
                        next
                }
 
-        # perhaps we have common known name?
+               replace_requires();
+       }
 
-        # jpackages
-               sub(/^java-devel$/, "jdk", $2);
-               sub(/^log4j$/, "jakarta-log4j", $2);
-               sub(/^oro$/, "jakarta-oro", $2);
-               sub(/^xerces-j2$/, "xerces-j", $2);
-               sub(/^ant-junit$/, "jakarta-ant", $2);
-               sub(/^ldapjdk$/, "ldapsdk", $2);
-               sub(/^saxon-scripts$/, "saxon", $2);
+       if (field ~ /^requires:/ || field ~ /^requires\(/) {
+               replace_requires();
        }
 
+
        # obsolete/unwanted tags
        if (field ~ /vendor:|packager:|distribution:|docdir:|prefix:|icon:|author:|author-email:|metadata-version:/) {
                next
@@ -753,11 +799,61 @@ preamble == 1 {
                $1 = "License:"
        }
 
+       # ease updating from debian .dsc
+       if (field ~ /homepage:/) {
+               $1 = "URL:"
+       }
+
+       # suse
+       if (field ~ /recommends:/) {
+               $1 = "Suggests:"
+       }
+
+       if ($3 == "==" && $1 !~ /%/) {
+               $3 = "="
+       }
+
        if (field ~ /license:/) {
                l = substr($0, index($0, $2));
                if (l == "Python Software Foundation License") {
                        l = "PSF"
                }
+               if (l == "Apache License 2.0" || l == "Apache 2.0" || l == "Apache License Version 2.0" || l == "Apache License, Version 2.0" || l == "Apache Software License v2") {
+                       l = "Apache v2.0"
+               }
+               if (l == "Apache Group License" || l == "Apache Software License" || l == "Apache License") {
+                       l = "Apache"
+               }
+               if (l == "Apache-style License" || l == "Apache-style Software License") {
+                       l = "Apache-like"
+               }
+               if (l == "Apache Software License 1.1" || l == "Apache 1.1") {
+                       l = "Apache v1.1"
+               }
+               if (l == "GPLv2") {
+                       l = "GPL v2"
+               }
+               if (l == "GPLv2+") {
+                       l = "GPL v2+"
+               }
+               if (l == "GPL v2 or later") {
+                       l = "GPL v2+"
+               }
+               if (l == "LGPL v2.0 only") {
+                       l = "LGPL v2"
+               }
+               if (l == "LGPLv2+") {
+                       l = "LGPL v2+"
+               }
+               if (l == "GPLv3") {
+                       l = "GPL v3"
+               }
+               if (l == "GPLv3+") {
+                       l = "GPL v3+"
+               }
+               if (l == "MPLv1.1") {
+                       l = "MPL v1.1"
+               }
                $0 = "License:\t" l;
        }
 
@@ -798,6 +894,9 @@ preamble == 1 {
        if (field ~ /^url:$/)
                $1 = "URL:"
 
+       if (field ~ /^patch/)
+               $1 = "Patch" substr(field, 6);
+
        if (field ~ /^description:$/)
                $1 = "\n%description\n"
 
@@ -865,19 +964,14 @@ preamble == 1 {
                        if (_snap) {
                                url[n] = fixedsub(_snap, "%{_snap}", url[n])
                        }
+                       if (subver) {
+                               url[n] = fixedsub(subver, "%{subver}", url[n])
+                       }
                }
                # assigning to $2 kills preamble formatting
                $2 = fixedsub(filename, url[n], $2)
 
-               # sourceforge urls
-               sub("[?]use_mirror=.*$", "", $2);
-               sub("[?]download$", "", $2);
-               sub("^http://prdownloads\.sourceforge\.net/", "http://dl.sourceforge.net/", $2)
-               sub("^http://download\.sf\.net/", "http://dl.sourceforge.net/", $2)
-
-               sub("^http://.*\.dl\.sourceforge\.net/", "http://dl.sourceforge.net/", $2)
-               sub("^http://dl\.sourceforge\.net/sourceforge/", "http://dl.sourceforge.net/", $2)
-               sub("^http://dl\.sf\.net/", "http://dl.sourceforge.net/", $2)
+               $2 = unify_url($2)
        }
 
 
@@ -906,13 +1000,19 @@ preamble == 1 {
 #
 # NOTES:
 # - mixing BR/R and anything else confuses this (all will be sorted together)
-#   so don't do that.
+#      so don't do that.
 # - comments leading the BR/R can not be associated,
-#   so don't adapterize when the BR/R are mixed with comments
-ENVIRON["SKIP_SORTBR"] != 1 && preamble == 1 && $0 ~ PREAMBLE_TAGS, $0 ~ PREAMBLE_TAGS {
+#      so don't adapterize when the BR/R are mixed with comments
+ENVIRON["SKIP_SORTBR"] != 1 && preamble == 1 && $0 ~ PREAMBLE_TAGS ":", $0 ~ PREAMBLE_TAGS ":"{
        if ($1 ~ /Pre[Rr]eq:/) {
                sub(/Pre[Rr]eq:/, "Requires:", $1);
        }
+       if ($1 == "BR:" ) {
+               $1 = "BuildRequires:"
+       }
+       if ($1 == "R:" ) {
+               $1 = "Requires:"
+       }
        format_preamble()
 #      kill_preamble_macros(); # breaks tabbing
 
@@ -971,8 +1071,16 @@ preamble == 1 {
 
 
 END {
-       if (do_not_touch_anything)
-               exit 0
+       if (do_not_touch_anything) {
+               exit(rc)
+       }
+
+       # TODO: need to output these in proper place
+       if (BR_count > 0) {
+               for (i = 0; i <= BR_count; i++) {
+                       print BR[i];
+               }
+       }
 
        close(changelog_file)
        while ((getline < changelog_file) > 0)
@@ -990,26 +1098,38 @@ END {
                print "%define date\t%(echo `LC_ALL=\"C\" date +\"%a %b %d %Y\"`)"
        }
 
-       if (has_changelog == 0)
+       if (has_changelog == 0) {
                print "%changelog"
+       }
 
-       if (boc > 2)
+       if (boc > 2) {
                print "* %{date} PLD Team <feedback@pld-linux.org>"
+       }
        if (boc > 1) {
                printf "All persons listed below can be reached at "
                print "<cvs_login>@pld-linux.org\n"
        }
-       if (boc > 0)
+       if (boc > 0) {
                print "$" "Log:$"
+       }
 }
 
-function fixedsub(s1,s2,t, ind) {
 # substitutes fixed strings (not regexps)
+function fixedsub(s1,s2,t, ind) {
        if (ind = index(t,s1))
                t = substr(t, 1, ind-1) s2 substr(t, ind+length(s1))
        return t
 }
 
+# replace s with s2 if it equals to s1
+function replace(s, s1, s2) {
+       if (s == s1) {
+               return s2;
+       } else {
+               return s;
+       }
+}
+
 # There should be one or two tabs after the colon.
 function format_preamble()
 {
@@ -1017,11 +1137,12 @@ function format_preamble()
                return;
        }
        sub(/:[ \t]*/, ":")
-       if (match($0, /[A-Za-z0-9(),#_ \t]+[ \t]*:[ \t]*/) == 1) {
-               if (RLENGTH < 8)
+       if (match($0, /[A-Za-z0-9(),#_ \t.-]+[ \t]*:[ \t]*/) == 1) {
+               if (RLENGTH < 8) {
                        sub(/:/, ":\t\t")
-               else
+               } else {
                        sub(/:/, ":\t")
+               }
        }
 }
 
@@ -1038,6 +1159,12 @@ function use_macros()
                return;
        }
 
+       sub("%{_defaultdocdir}", "%{_docdir}");
+       sub("%{_bindir}/perl", "%{__perl}");
+       sub("%{_bindir}/python", "%{__python}");
+
+       gsub(infodir, "%{_infodir}")
+
        gsub(perl_sitearch, "%{perl_sitearch}")
        gsub(perl_archlib, "%{perl_archlib}")
        gsub(perl_privlib, "%{perl_privlib}")
@@ -1048,10 +1175,30 @@ function use_macros()
        gsub(py_sitescriptdir, "%{py_sitescriptdir}")
        gsub(py_sitedir, "%{py_sitedir}")
        gsub(py_scriptdir, "%{py_scriptdir}")
-       gsub("%{_libdir}/python2.4/site-packages", "%{py_sitedir}")
-       gsub("%{python_sitelib}", "%{py_sitedir}")
+
+       gsub(py3_sitescriptdir, "%{py3_sitescriptdir}")
+       gsub(py3_sitedir, "%{py3_sitedir}")
+       gsub(py3_scriptdir, "%{py3_scriptdir}")
+
+       gsub(ruby_archdir, "%{ruby_archdir}")
+       gsub(ruby_ridir, "%{ruby_ridir}")
+       gsub(ruby_rubylibdir, "%{ruby_rubylibdir}")
+       gsub(ruby_sitearchdir, "%{ruby_sitearchdir}")
+       gsub(ruby_sitelibdir, "%{ruby_sitelibdir}")
+       gsub(ruby_rdocdir, "%{ruby_rdocdir}")
+
        gsub("%{_datadir}/applications", "%{_desktopdir}")
        gsub("%{_datadir}/pixmaps", "%{_pixmapsdir}")
+       gsub("%{_datadir}/java", "%{_javadir}")
+
+       gsub("%{_libdir}/pkgconfig", "%{_pkgconfigdir}")
+       gsub(pkgconfigdir, "%{_pkgconfigdir}")
+
+       gsub("%{_datadir}/pkgconfig", "%{_npkgconfigdir}")
+       gsub(npkgconfigdir, "%{_npkgconfigdir}")
+
+       gsub(libdir, "%{_libdir}")
+       gsub(javadir, "%{_javadir}")
 
        gsub(bindir, "%{_bindir}")
        gsub("%{prefix}/bin", "%{_bindir}")
@@ -1063,6 +1210,8 @@ function use_macros()
                        continue;
                if ($c ~ sbindir "/webapp")
                        continue;
+               if ($c ~ sbindir "/ldconfig")
+                       continue;
                if ($c ~ sbindir "/chsh")
                        continue;
                if ($c ~ sbindir "/usermod")
@@ -1088,7 +1237,9 @@ function use_macros()
                        continue;
                if ($c ~ sysconfdir "/{?modprobe.(d|conf)")
                        continue;
-               if ($c ~ sysconfdir "/{?udev/rules.d")
+               if ($c ~ sysconfdir "/{?udev")
+                       continue;
+               if ($c ~ sysconfdir "/{?hotplug")
                        continue;
                if ($c ~ sysconfdir "/{?logrotate.d")
                        continue;
@@ -1118,13 +1269,36 @@ function use_macros()
                        continue;
                if ($c ~ sysconfdir "/{?samba")
                        continue;
+               if ($c ~ sysconfdir "/{?xdg")
+                       continue;
                if ($c ~ sysconfdir "/shells")
                        continue;
+               if ($c ~ sysconfdir "/inittab")
+                       continue;
+               if ($c ~ sysconfdir "/init")
+                       continue;
+               if ($c ~ sysconfdir "/ppp")
+                       continue;
+               if ($c ~ sysconfdir "/dbus-1")
+                       continue;
+               if ($c ~ sysconfdir "/tmpwatch")
+                       continue;
+               if ($c ~ sysconfdir "/acpi")
+                       continue;
+               if ($c ~ sysconfdir "/apm")
+                       continue;
                gsub(sysconfdir, "%{_sysconfdir}", $c)
        }
 
        gsub(docdir, "%{_docdir}")
+
+       gsub(kdedocdir, "%{_kdedocdir}")
+
+       gsub(gtkdocdir, "%{_gtkdocdir}")
+       gsub("%{_docdir}/gtk-doc/html", "%{_gtkdocdir}")
+
        gsub(php_pear_dir, "%{php_pear_dir}")
+       gsub(php_data_dir, "%{php_data_dir}")
 
        for (c = 1; c <= NF; c++) {
                if ($c ~ datadir "/automake")
@@ -1215,6 +1389,15 @@ function use_macros()
        gsub("^make$", "%{__make}")
        gsub("^make ", "%{__make} ")
        gsub("^gcc ", "%{__cc} ")
+       gsub("^rm --interactive=never ", "%{__rm} ")
+
+       # fedora
+       gsub("%{ruby_sitearch}", "%{ruby_sitearchdir}")
+       gsub("%{python_sitearch}", "%{py_sitedir}")
+       gsub("%{python_sitelib}", "%{py_sitescriptdir}")
+
+       # alt linux
+       gsub("%_man1dir", "%{_mandir}/man1")
 
        # mandrake specs
        gsub("^%make$", "%{__make}")
@@ -1224,18 +1407,20 @@ function use_macros()
        gsub("^%makeinstall", "%{__make} install \\\n\tDESTDIR=$RPM_BUILD_ROOT")
        gsub("^%{__rm} -rf %{buildroot}", "rm -rf $RPM_BUILD_ROOT")
        gsub("^%{__install}", "install")
-       gsub("^%{__rm}", "rm")
        gsub("%optflags", "%{rpmcflags}")
        gsub("%{compat_perl_vendorarch}", "%{perl_vendorarch}")
 
        gsub("^%{__make} install DESTDIR=\$RPM_BUILD_ROOT", "%{__make} install \\\n\tDESTDIR=$RPM_BUILD_ROOT")
-       gsub("^fix-info-dir$", "[ ! -x /usr/sbin/fix-info-dir ] || /usr/sbin/fix-info-dir -c %{_infodir} >/dev/null 2>\&1")
+       gsub("^fix-info-dir$", "[ ! -x /usr/sbin/fix-info-dir ] || /usr/sbin/fix-info-dir -c %{_infodir} >/dev/null 2>\\&1")
        $0 = fixedsub("%buildroot", "$RPM_BUILD_ROOT", $0)
        $0 = fixedsub("%{buildroot}", "$RPM_BUILD_ROOT", $0)
        $0 = fixedsub("CXXFLAGS=%{rpmcflags} %configure", "CXXFLAGS=%{rpmcflags}\n%configure", $0);
+       $0 = fixedsub("%__install", "install", $0);
 
        # split configure line to multiple lines
-       if (/%configure / && !/\\$/) {
+       if (/%configure +$/) {
+               sub( / +$/, "" );
+       } else if (/%configure / && !/\\$/) {
                $0 = format_configure($0);
        }
 
@@ -1245,17 +1430,35 @@ function use_macros()
        gsub("%_sbindir", "%{_sbindir}")
        gsub("%_mandir", "%{_mandir}")
        gsub("%name", "%{name}")
-    gsub(/%ant/, "ant")
-    gsub(/%__rm/, "rm");
-    gsub(/%__mkdir_p/, "install -d");
-    gsub(/%__cp/, "cp");
-    gsub(/%__ln_s/, "ln -s");
-    gsub(/%__sed/, "%{__sed}");
-    gsub(/%__cat/, "cat");
-    gsub(/%__chmod/, "chmod");
+       gsub(/%__rm/, "rm");
+       gsub(/%__mkdir_p/, "install -d");
+       gsub(/%__cp/, "cp");
+       gsub(/%__ln_s/, "ln -s");
+       gsub(/%__sed/, "%{__sed}");
+       gsub(/%__cat/, "cat");
+       gsub(/%__chmod/, "chmod");
 
        gsub("/usr/src/linux", "%{_kernelsrcdir}")
        gsub("%{_prefix}/src/linux", "%{_kernelsrcdir}")
+
+       if (/^ant / || /^%{ant}/) {
+               sub(/^ant/, "%ant")
+               sub(/^%{ant}/, "%ant")
+               add_br("BuildRequires:  jpackage-utils");
+               add_br("BuildRequires:  rpmbuild(macros) >= 1.300");
+       }
+
+       $0 = fixedsub("%(%{__cc} -dumpversion)", "%{cc_version}", $0);
+       $0 = fixedsub("%(%{__cxx} -dumpversion)", "%{cxx_version}", $0);
+
+       # kill the _class and _subclass pear macros
+       if (pear_class) {
+               gsub("%{_class}", pear_class);
+       }
+       if (pear_subclass) {
+               gsub("%{_subclass}", pear_subclass);
+       }
+
 }
 
 function format_configure(line,                n, a, s) {
@@ -1282,13 +1485,13 @@ function isort(A,n,             i,j,hold) {
 }
 
 
-function use_files_macros(     i, n, t, a)
+function use_files_macros(     i, n, t, a, l)
 {
        use_macros()
 
        # skip comments
        if (/^#/) {
-               return;
+               return 1;
        }
 
        sub("^%doc %{_mandir}", "%{_mandir}")
@@ -1318,6 +1521,9 @@ function use_files_macros(        i, n, t, a)
        gsub("%{_sysconfdir}/sysconfig", "/etc/sysconfig")
        gsub("%{_sysconfdir}/certs", "/etc/certs")
        gsub("%{_sysconfdir}/init.d", "/etc/init.d")
+       gsub("%{_sysconfdir}/dbus-1", "/etc/dbus-1")
+       gsub("%{_sysconfdir}/pki", "/etc/pki")
+       gsub("%{_sysconfdir}/tmpwatch", "/etc/tmpwatch")
 
        # /etc/init.d -> /etc/rc.d/init.d
        if (!/^\/etc\/init\.d$/) {
@@ -1333,7 +1539,7 @@ function use_files_macros(        i, n, t, a)
                }
        }
 
-       if (/lib.+\.so/ && !/\.so$/ && !/^%attr.*/ && !/%exclude/) {
+       if (/lib.+\.so\b/ && !/\.so$/ && !/^%attr.*/ && !/%exclude/) {
                $0 = "%attr(755,root,root) " $0
        }
 
@@ -1341,15 +1547,20 @@ function use_files_macros(      i, n, t, a)
                $0 = "%attr(755,root,root) " $0
        }
 
+       # remove attrs from man pages
+       if (/%{_mandir}/ && /^%attr/) {
+               sub("^%attr\\(.*\\) *", "");
+       }
+
        # /etc/sysconfig files
        # %attr(640,root,root) %config(noreplace) %verify(not size mtime md5) /etc/sysconfig/*
        # attr not required, allow default 644 attr
        if (!/network-scripts/ && !/%dir/ && !/\.d$/ && !/functions/ && !/\/etc\/sysconfig\/wmstyle/) {
-               if (/\/etc\/sysconfig\// && /%config/ && !/%config\(noreplace\)/) {
+               if (/\/etc\/sysconfig\// && /%config/ && !/%config\(noreplace/) {
                        gsub("%config", "%config(noreplace)")
                }
 
-               if (/\/etc\/sysconfig\// && !/%config\(noreplace\)/) {
+               if (/\/etc\/sysconfig\// && !/%config\(noreplace/) {
                        $NF = "%config(noreplace) " $NF
                }
 
@@ -1367,11 +1578,15 @@ function use_files_macros(      i, n, t, a)
                gsub("%attr\\(0", "%attr(")
        }
 
-    # kill default attrs
-    gsub(/%dir %attr\(755,root,root\)/, "%dir");
-    if (!/%dir/) {
-        gsub(/%attr\(644,root,root\)/, "");
-    }
+       # kill leading whitespace
+       gsub(/^ +/, "");
+
+       # kill default attrs
+       gsub(/%dir %attr\(755,root,root\)/, "%dir");
+       gsub(/%attr\(755,root,root\) %dir/, "%dir");
+       if (!/%dir/) {
+               gsub(/%attr\(644,root,root\) /, "");
+       }
 
        # sort %verify attrs
        if (match($0, /%verify\(not([^)]+)\)/)) {
@@ -1394,6 +1609,7 @@ function use_files_macros(        i, n, t, a)
 
        if (/%{_mandir}/) {
                gsub("\.gz$", "*")
+               gsub("%ext_man$", "*")
        }
 
        # locale dir and no %lang -> bad
@@ -1401,6 +1617,29 @@ function use_files_macros(       i, n, t, a)
                $(NF + 1) = "# FIXME consider using %find_lang"
        }
 
+       # python egg-infos
+       if (match($0, "^%{py_site(script)?dir}/.+-py"py_ver".egg-info$")) {
+               # tests:
+               #%{py_sitedir}/*-py2.4.egg-info
+               #%{py_sitescriptdir}/GnuPGInterface-%{version}-py2.4.egg-info
+               #%{py_sitescriptdir}/python_mpd-%{version}-py2.4.egg-info
+               #%{py_sitescriptdir}/mechanize-0.1.6b-py2.4.egg-info
+
+               l = index($0, "/");
+               t = substr($0, 0, l);
+               s = substr($0, l + 1, RLENGTH - l - length("-py"py_ver".egg-info"));
+               if (match(s, "[^-]+$")) {
+                       if (RSTART > 1) {
+                               s = substr(s, 0, RSTART - 1);
+                       }
+                       print "%if \"%{py_ver}\" > \"2.4\""
+                       gsub(t "/.+.egg-info", t "/" s "-*.egg-info");
+                       print
+                       print "%endif"
+                       return 0;
+               }
+       }
+
        # atrpms
        $0 = fixedsub("%{perl_man1dir}", "%{_mandir}/man1", $0);
        $0 = fixedsub("%{perl_man3dir}", "%{_mandir}/man3", $0);
@@ -1413,6 +1652,10 @@ function use_files_macros(       i, n, t, a)
        gsub("%{_datadir}/applications", "%{_desktopdir}");
        gsub("%{_datadir}/icons", "%{_iconsdir}");
        gsub("%{_datadir}/pixmaps", "%{_pixmapsdir}");
+       gsub("%{_datadir}/pear", "%{php_pear_dir}");
+       gsub("%{_datadir}/php", "%{php_data_dir}");
+
+       return 1
 }
 
 function use_script_macros()
@@ -1472,11 +1715,78 @@ function cflags(var)
        return 1
 }
 
+# return whole matched pattern
+function matchstr(str, pat)
+{
+       match(str, "[^/]+$");
+       return substr(str, RSTART, RLENGTH);
+}
+
+function unify_url(url)
+{
+
+       # sourceforge urls
+       # Docs about sourceforge mirror system: http://sourceforge.net/apps/trac/sourceforge/wiki/Mirrors
+
+       # 1. unify domains
+       sub("^http://prdownloads\.sourceforge\.net/", "http://downloads.sourceforge.net/", url)
+       sub("^http://download\.sf\.net/", "http://downloads.sourceforge.net/", url)
+       sub("^http://download\.sourceforge\.net/", "http://downloads.sourceforge.net/", url)
+       sub("^http://dl\.sourceforge\.net/", "http://downloads.sourceforge.net/", url)
+       sub("^http://.*\.dl\.sourceforge\.net/", "http://downloads.sourceforge.net/", url)
+       sub("^http://dl\.sf\.net/", "http://downloads.sourceforge.net/", url)
+       sub("^http://downloads\.sourceforge\.net/sourceforge/", "http://downloads.sourceforge.net/", url)
+
+       # 3. unify urls
+       if (url ~ /sourceforge.net/) {
+               sub("[?&]big_mirror=.*$", "", url);
+               sub("[?&]modtime=.*$", "", url);
+               sub("[?]use_mirror=.*$", "", url);
+               sub("[?]download$", "", url);
+               sub("/download$", "", url);
+       }
+
+       # SF: new style urls, strip "files/" between and prepend dl.
+       if (match(url, "^http://sourceforge.net/projects/[^/]+/files/")) {
+               url = substr(url, 1, RLENGTH - length("files/")) substr(url, RSTART + RLENGTH);
+               sub("^http://sourceforge.net/projects/", "http://downloads.sourceforge.net/project/", url);
+       }
+
+       # SF unify: http://downloads.sourceforge.net/PROJECT/TARBALL
+       # http://downloads.sourceforge.net/project/PROJECT/FILE/VERSION/%{name}-%{version}.zip
+       if (match(url, "^http://downloads.sourceforge.net/project/[^/]+")) {
+               url = sprintf("http://downloads.sourceforge.net/%s/%s", substr(url, 42, RLENGTH - 41), matchstr(url, "[^/]+$"));
+       }
+
+       sub("^ftp://ftp\.gnome\.org/", "http://ftp.gnome.org/", url)
+       sub("^http://ftp\.gnome\.org/pub/gnome/", "http://ftp.gnome.org/pub/GNOME/", url)
+
+       # apache urls
+       sub("^http://apache.zone-h.org/", "http://www.apache.org/dist/", url)
+
+       # gnu.org
+       sub("^ftp://ftp\.gnu\.org/", "http://ftp.gnu.org/", url)
+       sub("^http://ftp\.gnu\.org/pub/gnu/", "http://ftp.gnu.org/gnu/", url)
+
+       # debian.org
+       sub("^ftp://ftp\.[^.]+\.debian\.org/", "ftp://ftp.debian.org/", url)
+       sub("^http://ftp\.[^.]+\.debian\.org/", "ftp://ftp.debian.org/", url)
+       sub("^ftp://ftp\.debian\.org/pub/debian/", "ftp://ftp.debian.org/debian/", url)
+
+       return url
+}
+
 function demacroize(str)
 {
        if (mod_name) {
                sub("%{mod_name}", mod_name, str);
        }
+       if (pearname) {
+               sub("%{_pearname}", pearname, str);
+       }
+       if (pearname) {
+               sub("%{pearname}", pearname, str);
+       }
        if (name) {
                sub("%{name}", name, str);
        }
@@ -1495,16 +1805,32 @@ function demacroize(str)
        if (_snap) {
                sub("%{_snap}", _snap, str);
        }
+       if (subver) {
+               sub("%{subver}", subver, str);
+       }
        return str;
 }
 
 function kill_preamble_macros()
 {
-       if ($1 ~ /^URL:/ || $1 ~ /^Obsoletes:/) {
+       if ($1 ~ /^Obsoletes:/) {
+               # NB! assigning $2 a value breaks tabbing
+               $2 = demacroize($2);
+       }
+       if ($1 ~ /^URL:/) {
                # NB! assigning $2 a value breaks tabbing
                $2 = demacroize($2);
-               # unify sourceforge url
-               sub("\.sf\.net/$", ".sourceforge.net/", $2);
+               $2 = unify_url($2)
+       }
+
+       # fedora extras
+       if (/%{\?FE_USERADD_REQ}/) {
+               $0 = "";
+               print "BuildRequires:   rpmbuild(macros) >= 1.202"
+               print "Provides:        user(xxx)"
+               print "Requires(postun):        /usr/sbin/userdel"
+               print "Requires(pre):  /bin/id"
+               print "Requires(pre):  /usr/sbin/useradd"
        }
 }
 
@@ -1553,3 +1879,383 @@ function use_tabs()
        # reverse vim: ts=4 sw=4 et
        gsub(/    /, "\t");
 }
+
+function add_br(br)
+{
+       BR[BR_count++] = br
+}
+
+# Load rpm macros
+# you should update the list also in adapter when making changes here
+function import_rpm_macros() {
+       # File with rpm groups
+       topdir = ENVIRON["_topdir"]
+
+       if (!topdir) {
+               print "adapter.awk should not not be invoked directly, but via adapter script" > "/dev/stderr"
+               do_not_touch_anything = 1
+               exit(rc = 1);
+       }
+
+       # update this version dep each time some new macro export is added
+       if (!ENVIRON["ADAPTER_REVISION"] || ENVIRON["ADAPTER_REVISION"] < 1.47) {
+               print "adapter shell script is outdated, please cvs up it" > "/dev/stderr"
+               do_not_touch_anything = 1
+               exit(rc = 1);
+       }
+
+       # get cvsaddress for changelog section
+       # using rpm macros as too lazy to add ~/.adapterrc parsing support.
+       _cvsmaildomain = ENVIRON["_cvsmaildomain"]
+       _cvsmailfeedback = ENVIRON["_cvsmailfeedback"]
+
+       prefix = ENVIRON["_prefix"]
+       bindir = ENVIRON["_bindir"]
+       sbindir = ENVIRON["_sbindir"]
+       libdir = ENVIRON["_libdir"]
+       sysconfdir = ENVIRON["_sysconfdir"]
+       datadir = ENVIRON["_datadir"]
+       includedir = ENVIRON["_includedir"]
+       mandir = ENVIRON["_mandir"]
+       infodir = ENVIRON["_infodir"]
+       examplesdir = ENVIRON["_examplesdir"]
+       docdir = ENVIRON["_defaultdocdir"]
+       kdedocdir = ENVIRON["_kdedocdir"]
+       gtkdocdir = ENVIRON["_gtkdocdir"]
+       desktopdir = ENVIRON["_desktopdir"]
+       pixmapsdir = ENVIRON["_pixmapsdir"]
+       javadir = ENVIRON["_javadir"]
+       pkgconfigdir = ENVIRON["_pkgconfigdir"]
+       npkgconfigdir = ENVIRON["_npkgconfigdir"]
+
+       perl_sitearch = ENVIRON["perl_sitearch"]
+       perl_archlib = ENVIRON["perl_archlib"]
+       perl_privlib = ENVIRON["perl_privlib"]
+       perl_vendorlib = ENVIRON["perl_vendorlib"]
+       perl_vendorarch = ENVIRON["perl_vendorarch"]
+       perl_sitelib = ENVIRON["perl_sitelib"]
+
+       py_sitescriptdir = ENVIRON["py_sitescriptdir"]
+       py_sitedir = ENVIRON["py_sitedir"]
+       py_scriptdir = ENVIRON["py_scriptdir"]
+       py_ver = ENVIRON["py_ver"]
+
+       py3_sitescriptdir = ENVIRON["py3_sitescriptdir"]
+       py3_sitedir = ENVIRON["py3_sitedir"]
+       py3_scriptdir = ENVIRON["py3_scriptdir"]
+       py3_ver = ENVIRON["py3_ver"]
+
+       ruby_archdir = ENVIRON["ruby_archdir"]
+       ruby_ridir = ENVIRON["ruby_ridir"]
+       ruby_rubylibdir = ENVIRON["ruby_rubylibdir"]
+       ruby_sitearchdir = ENVIRON["ruby_sitearchdir"]
+       ruby_sitelibdir = ENVIRON["ruby_sitelibdir"]
+       ruby_rdocdir = ENVIRON["ruby_rdocdir"]
+
+       php_pear_dir = ENVIRON["php_pear_dir"]
+       php_data_dir = ENVIRON["php_data_dir"]
+       tmpdir = ENVIRON["tmpdir"]
+}
+
+# php virtual deps as discussed in devel-en
+function replace_php_virtual_deps() {
+       pkg = $2
+#      if (pkg == "php-program") {
+#              $0 = $1 "\t/usr/bin/php"
+#              return
+#      }
+
+#      if (pkg ~ /^php-[a-z]/ && pkg !~ /^php-(pear|common|cli|devel|fcgi|cgi|dirs|program|pecl-)/) {
+#              sub(/^php-/, "php(", pkg);
+#              sub(/$/, ") # verify this correctness -- it may be wanted to use specific not virtual dep", pkg);
+#              $2 = pkg
+#      }
+
+       if (pkg ~/^php$/) {
+               $2 = "webserver(php)";
+               if ($4 ~ /^[0-9]:/) {
+                       $4 = substr($4, 3);
+               }
+       }
+
+       if (pkg ~/^php4$/) {
+               $2 = "webserver(php)";
+               if ($4 ~ /^[0-9]:/) {
+                       $4 = substr($4, 3);
+               }
+       }
+}
+
+function replace_groupnames(group) {
+       group = replace(group, "Amusements/Games", "Applications/Games");
+       group = replace(group, "Amusements/Games/Strategy/Real Time", "X11/Applications/Games/Strategy");
+       group = replace(group, "Application/Multimedia", "Applications/Multimedia");
+       group = replace(group, "Application/System", "Applications/System");
+       group = replace(group, "Applications/Compilers", "Development/Languages");
+       group = replace(group, "Applications/Daemons", "Daemons");
+       group = replace(group, "Applications/Internet", "Applications/Networking");
+       group = replace(group, "Applications/Internet/Peer to Peer", "Applications/Networking");
+       group = replace(group, "Applications/Productivity", "X11/Applications");
+       group = replace(group, "Applications/Security", "Applications/System");
+       group = replace(group, "Applications/Web", "Applications/WWW");
+       group = replace(group, "Database", "Applications/Databases");
+       group = replace(group, "Development/C", "Development/Libraries");
+       group = replace(group, "Development/Code Generators", "Development");
+       group = replace(group, "Development/Docs", "Documentation");
+       group = replace(group, "Development/Documentation", "Documentation");
+       group = replace(group, "Development/Java", "Development/Languages/Java");
+       group = replace(group, "Development/Languages/C and C++", "Libraries");
+       group = replace(group, "Development/Languages/Other", "Development/Languages");;
+       group = replace(group, "Development/Languages/Ruby", "Development/Languages");
+       group = replace(group, "Development/Libraries/C and C++", "Development/Libraries");
+       group = replace(group, "Development/Libraries/Java", "Development/Languages/Java");
+       group = replace(group, "Development/Libraries/Python", "Development/Languages/Python");
+       group = replace(group, "Development/Libraries/TCL", "Development/Languages/Tcl");;
+       group = replace(group, "Development/Other", "Development");
+       group = replace(group, "Development/Python", "Development/Languages/Python");
+       group = replace(group, "Development/Testing", "Development");
+       group = replace(group, "Editors", "Applications/Text");
+       group = replace(group, "Emulators", "Applications/Emulators");
+       group = replace(group, "File tools", "Applications/File");
+       group = replace(group, "Games", "Applications/Games");
+       group = replace(group, "Library/Development", "Development/Libraries");
+       group = replace(group, "Networking/Deamons", "Networking/Daemons");
+       group = replace(group, "Networking/Mail", "Applications/Mail");
+       group = replace(group, "Networking/Other", "Networking");
+       group = replace(group, "Productivity/Databases/Servers", "Applications/Databases");
+       group = replace(group, "Productivity/Multimedia/Other", "X11/Applications/Multimedia");
+       group = replace(group, "Productivity/Networking/Web/Servers", "Networking/Daemons/HTTP");;
+       group = replace(group, "Shells", "Applications/Shells");
+       group = replace(group, "System Environment/Base", "Base");
+       group = replace(group, "System Environment/Daemons", "Daemons");
+       group = replace(group, "System Environment/Kernel", "Base/Kernel");
+       group = replace(group, "System Environment/Libraries", "Libraries");
+       group = replace(group, "System Tools", "Applications/System");
+       group = replace(group, "System", "Base");
+       group = replace(group, "System/Base", "Base");
+       group = replace(group, "System/Kernel and hardware", "Base/Kernel");
+       group = replace(group, "System/Libraries", "Libraries");
+       group = replace(group, "System/Servers", "Daemons");
+       group = replace(group, "Text Processing/Markup/HTML", "Applications/Text");
+       group = replace(group, "Text Processing/Markup/XML", "Applications/Text");
+       group = replace(group, "Text tools", "Applications/Text");
+       group = replace(group, "User Interface/Desktops", "X11/Applications");
+       group = replace(group, "Utilities/System", "Applications/System");
+       group = replace(group, "Web/Database", "Applications/WWW");
+       group = replace(group, "X11/GNOME", "X11/Applications");
+       group = replace(group, "X11/GNOME/Applications", "X11/Applications");
+       group = replace(group, "X11/GNOME/Development/Libraries", "X11/Development/Libraries");
+       group = replace(group, "X11/Games", "X11/Applications/Games");
+       group = replace(group, "X11/Games/Strategy", "X11/Applications/Games/Strategy");
+       group = replace(group, "X11/Library", "X11/Libraries");
+       group = replace(group, "X11/Utilities", "X11/Applications");
+       group = replace(group, "X11/XFree86", "X11");
+       group = replace(group, "X11/Xserver", "X11/Servers");
+
+       return group;
+}
+
+function replace_requires() {
+
+       sub(/^python-setuptools-devel$/, "python-distribute", $2);
+       sub(/^gcc-g77/, "gcc-fortran", $2);
+
+       # use virtual, not package name
+       sub(/^rpm-build-macros$/, "rpmbuild(macros)", $2);
+
+       # bad package.xml, see http://pear.php.net/bugs/bug.php?id=17779
+       sub(/^php-php-gtk/, "php-gtk2", $2);
+
+       # jpackages
+       sub(/^antlr3$/, "java-antlr3", $2);
+       sub(/^avalon-framework$/, "java-avalon-framework", $2);
+       sub(/^avalon-logkit$/, "java-avalon-logkit", $2);
+       sub(/^axis$/, "java-axis", $2);
+       sub(/^bsf$/, "java-bsf", $2);
+       sub(/^gnu-regexp$/, "java-gnu-regexp", $2);
+       sub(/^gnu.regexp$/, "java-gnu-regexp", $2);
+       sub(/^hamcrest$/, "java-hamcrest", $2);
+       sub(/^jaas$/, "java(jaas)", $2);
+       sub(/^jaf$/, "java(jaf)", $2);
+       sub(/^jakarta-ant$/, "ant", $2);
+       sub(/^jakarta-commons-httpclient$/, "java-commons-httpclient", $2);
+       sub(/^jakarta-log4j$/, "java-log4j", $2);
+       sub(/^jakarta-oro$/, "java-oro", $2);
+       sub(/^jakarta-servletapi$/, "java(servlet)", $2);
+       sub(/^java-devel$/, "jdk", $2);
+       sub(/^java\(JSP\)$/, "java(jsp)", $2);
+       sub(/^java\(JavaServerFaces\)$/, "java(javaserverfaces)", $2);
+       sub(/^java\(Portlet\)$/, "java(portlet)", $2);
+       sub(/^java\(Servlet\)$/, "java(servlet)", $2);
+       sub(/^javamail$/, "java(javamail)", $2);
+       sub(/^jaxp$/, "java(jaxp)", $2);
+       sub(/^jaxp_parser_impl$/, "java(jaxp_parser_impl)", $2);
+       sub(/^jaxp_transform_impl$/, "java(jaxp_transform_impl)", $2);
+       sub(/^jce$/, "java(jce)", $2);
+       sub(/^jcommon$/, "java-jcommon", $2);
+       sub(/^jdbc-stdext$/, "java(jdbc-stdext)", $2);
+       sub(/^jdepend$/, "java-jdepend", $2);
+       sub(/^jfreechart$/, "java-jfreechart", $2);
+       sub(/^jmx$/, "java(jmx)", $2);
+       sub(/^jndi$/, "java(jndi)", $2);
+       sub(/^jsch$/, "java-jsch", $2);
+       sub(/^jsse$/, "java(jsse)", $2);
+       sub(/^jta$/, "java(jta)", $2);
+       sub(/^junit$/, "java-junit", $2);
+       sub(/^ldapjdk$/, "ldapsdk", $2);
+       sub(/^log4j$/, "java-log4j", $2);
+       sub(/^logging-log4j$/, "java-log4j", $2);
+       sub(/^oro$/, "java-oro", $2);
+       sub(/^rhino$/, "java-rhino", $2);
+       sub(/^saxon-scripts$/, "saxon", $2);
+       sub(/^servlet$/, "java(servlet)", $2);
+       sub(/^uddi4j$/, "java-uddi4j", $2);
+       sub(/^wsdl4j$/, "java-wsdl4j", $2);
+       sub(/^xalan-j$/, "java-xalan", $2);
+       sub(/^xalan-j2$/, "java-xalan", $2);
+       sub(/^xerces-j$/, "java(jaxp_parser_impl)", $2);
+       sub(/^xerces-j2$/, "java(jaxp_parser_impl)", $2);
+       sub(/^java-xerces$/, "java(jaxp_parser_impl)", $2);
+       sub(/^xml-commons-apis$/, "java-xml-commons-apis", $2);
+       sub(/^xml-commons-resolver$/, "java-xml-commons-resolver", $2);
+
+       # fedora / redhat
+       sub(/^keyutils-libs-devel$/, "keyutils-devel", $2);
+       sub(/^Django$/, "python-django", $2);
+       sub(/^GitPython$/, "python-git", $2);
+       sub(/^PyQt4-devel$/, "python-PyQt4-devel", $2);
+       sub(/^PyQwt-devel$/, "python-PyQwt-devel", $2);
+       sub(/^ccid$/, "pcsc-driver-ccid", $2);
+       sub(/^chkconfig$/, "/sbin/chkconfig", $2);
+       sub(/^db4-devel$/, "db-devel", $2);
+       sub(/^dbus-python$/, "python-dbus", $2);
+       sub(/^file-devel$/, "libmagic-devel", $2);
+       sub(/^freetype2-devel$/, "freetype-devel", $2);
+       sub(/^fuse-devel$/, "libfuse-devel", $2);
+       sub(/^gamin-python$/, "python-gamin", $2);
+       sub(/^gcc-c\+\+$/, "libstdc++-devel", $2);
+       sub(/^gnome-python2-extras$/, "python-gnome-extras", $2);
+       sub(/^gnome-python2-gconf$/, "python-gnome-gconf", $2);
+       sub(/^gnome-python2-gnomekeyring$/, "python-gnome-desktop-keyring", $2);
+       sub(/^gnome-python2-gtkspell$/, "python-gnome-extras-gtkspell", $2);
+       sub(/^gtk-sharp2-devel$/, "dotnet-gtk-sharp2-devel", $2);
+       sub(/^gtk2$/, "gtk+2", $2);
+       sub(/^gtk2-devel$/, "gtk+2-devel", $2);
+       sub(/^gtk3-devel$/, "gtk+3-devel", $2);
+       sub(/^initscripts$/, "rc-scripts", $2);
+       sub(/^iproute$/, "iproute2", $2);
+       sub(/^iscsi-initiator-utils$/, "open-iscsi", $2);
+       sub(/^jakarta-commons-logging$/, "java-commons-logging", $2);
+       sub(/^libXft-devel$/, "xorg-lib-libXft-devel", $2);
+       sub(/^libXrandr-devel$/, "xorg-lib-libXrandr-devel", $2);
+       sub(/^libacl-devel$/, "acl-devel", $2);
+       sub(/^libcurl-devel$/, "curl-devel", $2);
+       sub(/^libgudev1-devel$/, "udev-glib-devel", $2);
+       sub(/^libloader$/, "java-libloader", $2);
+       sub(/^libsrtp-devel$/, "srtp-devel", $2);
+       sub(/^libtdb$/, "tdb", $2);
+       sub(/^libtdb-devel$/, "tdb-devel", $2);
+       sub(/^libtevent$/, "tevent", $2);
+       sub(/^libtevent-devel$/, "tevent-devel", $2);
+       sub(/^mod_wsgi$/, "apache-mod_wsgi", $2);
+       sub(/^notify-python$/, "python-pynotify", $2);
+       sub(/^pcsc-lite-ccid$/, "pcsc-driver-ccid", $2);
+       sub(/^pulseaudio-libs-devel$/, "pulseaudio-devel", $2);
+       sub(/^pyOpenSSL$/, "python-pyOpenSSL", $2);
+       sub(/^pyflakes$/, "python-pyflakes", $2);
+       sub(/^pygobject2$/, "python-pygobject", $2);
+       sub(/^pygtk2$/, "python-pygtk", $2);
+       sub(/^pygtk2-devel$/, "python-pygtk-devel", $2);
+       sub(/^pygtk2-libglade$/, "python-pygtk-glade", $2);
+       sub(/^pysvn$/, "python-pysvn", $2);
+       sub(/^pytalloc$/, "python-talloc", $2);
+       sub(/^pytalloc-devel$/, "python-talloc-devel", $2);
+       sub(/^python-enchant$/, "python-pyenchant", $2);
+       sub(/^python-imaging$/, "python-PIL", $2);
+       sub(/^python-imaging-tk$/, "python-PIL-tk", $2);
+       sub(/^python-pygtk$/, "python-pygtk-gtk", $2);
+       sub(/^python-recaptcha-client$/, "python-recaptcha", $2);
+       sub(/^python-twisted-core$/, "python-TwistedCore", $2);
+       sub(/^python-twisted-core$/, "python-TwistedCore", $2);
+       sub(/^python-twisted-names$/, "python-TwistedNames", $2);
+       sub(/^python2-devel$/, "python-devel", $2);
+       sub(/^pytz$/, "python-pytz", $2);
+       sub(/^qt4-devel$/, "qt4-build", $2);
+       sub(/^qt4-webkit-devel$/, "QtWebKit-devel", $2);
+       sub(/^qtlockedfile-devel$/, "QtLockedFile-devel", $2);
+       sub(/^qtsingleapplication-devel$/, "QtSingleApplication-devel", $2);
+       sub(/^rpm-python$/, "python-rpm", $2);
+       sub(/^sip-devel$/, "python-sip-devel", $2);
+       sub(/^tftp-server$/, "tftpdaemon", $2);
+       sub(/^tkinter$/, "python-tkinter", $2);
+       sub(/^xapian-bindings-python$/, "python-xapian", $2);
+       sub(/^xorg-x11-server-sdk$/, "xorg-xserver-server-devel", $2);
+       # mandriva
+       sub(/^python-gobject-devel$/, "python-pygobject-devel", $2);
+       sub(/^python-pyrex$/, "python-Pyrex", $2);
+       sub(/^webkitgtk-devel$/, "gtk-webkit-devel", $2);
+       sub(/^python-curl$/, "python-pycurl", $2);
+       sub(/^python-webkitgtk$/, "python-pywebkitgtk", $2);
+       sub(/^pygtk2.0$/, "python-pygtk-gtk", $2);
+       sub(/^gnome-python-gconf$/, "python-gnome-gconf", $2);
+
+       # debian / ubuntu
+       sub(/^blkid-dev$/, "libblkid-devel", $2);
+       sub(/^ext2fs-dev$/, "e2fsprogs-devel", $2);
+       sub(/^libao-dev$/, "libao-devel", $2);
+       sub(/^libboost-filesystem[0-9.]+-dev$/, "boost-devel", $2);
+       sub(/^libboost-program-options[0-9.]+-dev$/, "boost-devel", $2);
+       sub(/^libboost-regex[0-9.]+-dev$/, "boost-devel", $2);
+       sub(/^libboost-thread[0-9.]+-dev$/, "boost-devel", $2);
+       sub(/^libcurl4-openssl-dev$/, "curl-devel", $2);
+       sub(/^libdnet-dev$/, "libdnet-devel", $2);
+       sub(/^libesd0-dev$/, "esound-devel", $2);
+       sub(/^libfishsound1-dev$/, "libfishsound-devel", $2);
+       sub(/^libgconf2-dev$/, "GConf2-devel", $2);
+       sub(/^libgl1-mesa-dev$/, "OpenGL-devel", $2);
+       sub(/^libgl1-mesa-dri$/, "OpenGL", $2);
+       sub(/^libglib2.0-dev$/, "glib2-devel", $2);
+       sub(/^libglu1-mesa-dev$/, "OpenGL-GLU-devel", $2);
+       sub(/^libgtk2.0-dev$/, "gtk+2-devel", $2);
+       sub(/^libhunspell-dev$/, "hunspell-devel", $2);
+       sub(/^libmcrypt-dev$/, "libmcrypt-devel", $2);
+       sub(/^libmhash-dev$/, "mhash-devel", $2);
+       sub(/^liboggz1-dev$/, "libggz-devel", $2);
+       sub(/^libpango1.0-dev$/, "pango-devel", $2);
+       sub(/^libqt4-dev$/, "qt4-build", $2);
+       sub(/^libshout3-dev$/, "libshout-devel", $2);
+       sub(/^libslp-dev$/, "openslp-devel", $2);
+       sub(/^libsndfile1-dev$/, "libsndfile-devel", $2);
+       sub(/^libspeex-dev$/, "speex-devel", $2);
+       sub(/^libssl-dev$/, "openssl-devel", $2);
+       sub(/^libvorbis-dev$/, "libvorbis-devel", $2);
+       sub(/^libxslt1-dev$/, "libxslt-devel", $2);
+       sub(/^libxss-dev$/, "xorg-lib-libXScrnSaver-devel", $2);
+       sub(/^mesa-common-dev$/, "OpenGL-devel", $2);
+       sub(/^libudev$/, "udev-libs", $2);
+       sub(/^tcp_wrappers-devel$/, "libwrap-devel", $2);
+       sub(/^vala-tools$/, "vala", $2);
+       sub(/^vala-devel$/, "vala", $2);
+
+       # altlinux
+       sub(/^libgit-devel$/, "git-core-devel", $2);
+       sub(/^libncurses-devel$/, "ncurses-devel", $2);
+       sub(/^libncursesxx-devel$/, "ncurses-c++-devel", $2);
+       sub(/^libpcre-devel$/, "pcre-devel", $2);
+       sub(/^libssl-devel$/, "openssl-devel", $2);
+
+       # suse
+       sub(/^alsa-devel$/, "alsa-lib-devel", $2);
+       sub(/^gtk-sharp2$/, "dotnet-gtk-sharp2", $2);
+       sub(/^gtkmm2-devel$/, "gtkmm-devel", $2);
+       sub(/^libexpat-devel$/, "expat-devel", $2);
+       sub(/^libffmpeg-devel$/, "ffmpeg-devel", $2);
+       sub(/^libopenssl-devel$/, "openssl-devel", $2);
+       sub(/^libpulse-devel$/, "pulseaudio-devel", $2);
+       sub(/^monodoc-core$/, "mono-monodoc", $2);
+       sub(/^python-gtk$/, "python-pygtk-gtk", $2);
+
+       replace_php_virtual_deps()
+}
+
+# vim:ts=4:sw=4
This page took 0.19942 seconds and 4 git commands to generate.