#!/usr/bin/perl
# -*- Mode: cperl -*-
#--------------------------------------------------------------------
# Copyright (C) 2000, 2001 by MandrakeSoft.
# Chmouel Boudjnah <chmouel@mandrakesoft.com>.
#
# Redistribution of this file is permitted under the terms of the GNU 
# Public License (GPL)
#--------------------------------------------------------------------
# $Id: grub,v 1.16 2001/10/05 13:19:15 chmouel Exp $
#--------------------------------------------------------------------
## description: 
#	      Add/check entry for grub bootloader.

my $debug = 0;

use strict;
use lib qw(/usr/share/loader);
use common;

my ($version, $title, $remove, $blank);
my $boot="/boot/";

while ( $ARGV[0] =~ /^-/ ) {
    $_ = shift;
    if (m/^-r/) {
	$remove++;
    } else {
	print STDERR "Unknow option $_";
    }
}

my $map_file = "/boot/grub/device.map";

my $options = common::getoptions();
# Try absolutely to get the root=
$options .= common::getroot();

unless ($version = shift) {
    print STDERR "No kernel version has been given using the current\n"; 
    $version = `uname -r`;chomp $version;
}
(my $glabel = $version) =~ s|\.||g; $glabel =~ s|mdk||; $glabel =~ s|secure|sec| if $glabel =~ /secure/; #compact entry
$glabel =~ s|enterprise|ent| if $glabel =~ /enterprise/;

my $cnt = 0;
my $grub_partition;

# we keep entry as hash for future use.
my (%main, %entry);
my $grub_conf = $ENV{GRUB_CONF} ? $ENV{GRUB_CONF} : "/boot/grub/menu.lst";

`cp -f $grub_conf ${grub_conf}.old` if -f $grub_conf && !$debug;

open F, $grub_conf or die "Can't open $grub_conf\n";
    
while (<F>) {
    $blank = "\n" if eof and $_ !~ /^\s*$/;

    next if /^\s*#/;
    $main{default} = $1 if /^default (\d+)/;
    
    if (/^title\s+(.*)/) {
	$title=$1;
	$entry{$title}{cnt} = $cnt;
	$main{default} = $title if $entry{$title}{cnt} == $main{default};
	$cnt++;
    }
    
    if (m/kernel\s+(\(.*\))([^ 	]+)\s+(.*)/) {
	$entry{$title}{label}=$title;
	$entry{$title}{partition} = $1;
	$entry{$title}{kernel} = $2;
	$entry{$title}{options} = $3;
    }
    $entry{$title}{initrd} = $1 if m/\s*initrd.*\)(.*)/;
}
close F;

common::check_default_entry(\%entry, \%main);
common::do_check($entry{$glabel}{label}, $version, $remove) unless $debug;

unless ($grub_partition = convert_grub_part(get_boot_partitions())) {
    die "Can't convert grub_partition\n";
}

if ($remove) {
    my $tmp = $debug ? "true" : "mktemp";
    my $output = `$tmp /tmp/.grub.XXXXXX`;chomp($output);
    local $/; local *F; local *O;
    
    if ( -l $entry{$main{default}}{kernel}) {
	my $resolved_link = readlink($entry{$main{default}}{kernel});
	my $type = common::get_kernel_type($entry{$main{default}}{kernel});

	# This should be broken :p
	if ($resolved_link =~ m|vmlinuz-$version|) {
	    
	    my $first_kernel = common::get_first_boot($type, $resolved_link);
	    
	    system("ln -fs vmlinuz-$first_kernel /boot/vmlinuz$type");
	    system("ln -fs initrd-$first_kernel.img /boot/initrd$type.img") if -f "/boot/initrd$type.img";
	
	}
    }
    open F, $grub_conf;
    open O, ">$output";
    select O unless $debug;
    while (<F>) {
	if (m@title $glabel\nkernel.*vmlinuz-.*?(?=(title|$))@s) {
 	    if (m@title $glabel\nkernel.*vmlinuz-.*title@s) {
 		$_ =~ s@title $glabel\nkernel.*vmlinuz-.*?(?=title)@@s;
 	    } else {
 		$_ =~ s@title $glabel\nkernel.*vmlinuz-.*$@@s;
 	    }
 	}
	print;
	
    }
    close F;
    select STDOUT;
    system("mv -f $output $grub_conf") unless $debug;
} else {
    unless ($debug) {
	open F, ">>$grub_conf" or die "Can't write to $grub_conf\n";
	select F;
    }
    print "\n" if $blank;
    print << "EOF";
title $glabel
kernel $grub_partition${boot}vmlinuz-$version $options
EOF
if ( -f "/boot/initrd-$version.img" ) {
    print "initrd $grub_partition${boot}initrd-$version.img\n";
}
close F;
select STDOUT;
}

sub convert_grub_part {
    my $fpart = shift;
    my ($disk, $part, $grub_part);
    do { $disk = $1; $part = $2 - 1 } if ($fpart =~ m|([^\d+]*)(\d+)|);
    local *F;
    open (F, $map_file) or die "Can't open $map_file\n";
    while (<F>) { $grub_part=$1  if (m|([^ \t]+)\s*$disk$|);}
    close F;
    $grub_part =~ s|\)|,$part\)|;
    return $grub_part =~ /\(([\w\d]+)d\d+,\d+\)/ ? $grub_part : undef;
}

sub get_boot_partitions {
    my $part;
    local *F;
    open F, '/etc/fstab'; 
    while (<F>) {
 	my @s = split ' ';
	$part = $s[0] if $s[1] =~ m|/$| and not $part;
	if ($s[1] =~ m|/boot$|) {
	    $boot="/";
	    $part = $s[0];
	}
    }; 
    close F;
    return  $part
}
