#!/usr/bin/perl -w

# Copyright (C) 1997,1998,1999, Roger Espel Llima
# Copyright (C) 2000, Sergey Babkin
# Copyright (C) 2009, Alex Kozlov
# 
# Permission is hereby granted, free of charge, to any person obtaining a copy
# of this software and any associated documentation files (the "Software"), to 
# deal in the Software without restriction, including without limitation the 
# rights to use, copy, modify, merge, publish, distribute, sublicense, 
# and/or sell copies of the Software, and to permit persons to whom the 
# Software is furnished to do so, subject to the following conditions:
# 
# The above copyright notice and this permission notice shall be included in
# all copies or substantial portions of the Software.
# 
# THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
# IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
# FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT.  IN NO EVENT SHALL THE
# SOFTWARE'S COPYRIGHT HOLDER(S) BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER 
# LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
# OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
# THE SOFTWARE

# (whew, that's done!)

# why does the world need another rpm2cpio?  because the existing one
# won't build unless you have half a ton of things that aren't really
# required for it, since it uses the same library used to extract RPM's.
# in particular, it won't build on the HPsUX box i'm on.
#
# added support for extracting to a specific directory

use strict;

my ($f, $p, $rpm, $filter) = ();

sub printhelp {
   print <<HERE;
rpm2cpio v1.3, perl version by orabidoo
unpacks the contents to dir

use: rpm2cpio file.rpm dir
HERE
   exit 0;
}

if ($#ARGV == 1) {
  open($f, "< $ARGV[0]") or die "Can't read file $ARGV[0]\n";
  $p = $ARGV[1];
  opendir(DIR, $p) || die "$p: $!";
  closedir(DIR);
} else {
  printhelp;
}

read $f, $rpm, 96;

my ($magic, $major, undef) = unpack("NCC", $rpm);

die "Not an RPM\n" if $magic != 0xedabeedb;
die "Not a version 3 or 4 RPM\n" if $major != 3 and $major != 4;

read $f, $rpm, 16 or die "No header\n";
while(1) {
   ($magic, undef, my $sections, my $bytes) = unpack("N4", $rpm);
   my ($smagic, $smagic2) = unpack("nN", $rpm);

   #printf(STDERR "0x%x 0x%x 0x%x 0x%x 0x%x\n",
   #  tell($f)-16, $magic, $sections, $bytes, $smagic);

   if ($smagic == 0x1f8b) {
      $filter = "gzip -cd";
      last;
   }
   # BZh
   if ($smagic == 0x425a and ($smagic2 & 0xff000000) == 0x68000000) {
      $filter = "bzip2 -cd";
      last;
   }
   # 0xFD, '7zXZ', 0x0
   if ($smagic == 0xfd37 and $smagic2 == 0x7a585a00) {
      $filter = "xz -cd";
      last;
   }
   # assume lzma if there is no sig
   if ($magic != 0x8eade801) {
      $filter = "lzma -cd";
      last;
   }

   # skip the headers
   seek $f, 16 * $sections + $bytes, 1 or die "File is too small\n";
   do {
      read $f, $rpm, 1 or die "No header\n" ;
   } while(0 == unpack("C", $rpm));
   read $f, $rpm, 15, 1 or die "No header\n" ;
}

chdir ($p);

open(ZCAT, "| $filter |cpio -d -i") or die "can't pipe to $filter\n";

while($rpm ne '') {
   print ZCAT $rpm;
   read $f, $rpm, 10240; # read in blocks
}

close ZCAT;
close $f;
