undo: clean up if only new_code loaded, too
[ksplice.git] / ksplice-obj.pl.in
bloba302e64ee000ce395d40d9094bec7e77a6f8addc
1 #!/usr/bin/perl
3 # Copyright (C) 2008-2009 Ksplice, Inc.
4 # Authors: Anders Kaseorg, Jeff Arnold, Tim Abbott
6 # This program is free software; you can redistribute it and/or modify
7 # it under the terms of the GNU General Public License, version 2.
9 # This program is distributed in the hope that it will be useful,
10 # but WITHOUT ANY WARRANTY; without even the implied warranty of
11 # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
12 # GNU General Public License for more details.
14 # You should have received a copy of the GNU General Public License
15 # along with this program; if not, write to the Free Software
16 # Foundation, Inc., 51 Franklin Street - Fifth Floor, Boston, MA
17 # 02110-1301, USA.
19 use strict;
20 use warnings;
21 use lib 'KSPLICE_DATA_DIR';
22 use Ksplice;
24 $Verbose::level = $ENV{KSPLICE_VERBOSE} if (defined $ENV{KSPLICE_VERBOSE});
26 sub empty_diff {
27 my ($out) = @_;
28 my ($obj) = $out =~ /^(.*)\.KSPLICE$/ or die;
29 unlink "$obj.KSPLICE_new_code" if (-e "$obj.KSPLICE_new_code");
30 unlink "$obj.KSPLICE_old_code" if (-e "$obj.KSPLICE_old_code");
31 open OUT, '>', "$out.tmp";
32 close OUT;
33 rename "$out.tmp", $out;
36 sub do_snap {
37 my ($out) = @_;
38 my ($obj) = $out =~ /^(.*)\.KSPLICE$/ or die;
39 die if (!-e $obj);
40 unlink "$obj.KSPLICE_pre" if (-e "$obj.KSPLICE_pre");
41 empty_diff($out);
44 sub do_diff {
45 my ($out) = @_;
46 my ($obj) = $out =~ /^(.*)\.KSPLICE$/ or die;
47 my $obj_pre = "$obj.KSPLICE_pre";
48 die if (!-e $obj);
49 die "Patch creates new object $obj" if (!-e $obj_pre);
50 if (system('cmp', '-s', '--', $obj_pre, $obj) == 0) {
51 unlink $obj_pre;
52 return empty_diff($out);
55 runval("$libexecdir/ksplice-objmanip", $obj, "$obj.KSPLICE_new_code", "keep-new-code", "$obj.KSPLICE_pre", $ENV{KSPLICE_KID});
56 return empty_diff($out) if (!-e "$obj.KSPLICE_new_code");
58 open OUT, '>', "$out.tmp";
59 print OUT "1\n";
60 close OUT;
61 rename "$out.tmp", $out;
63 runval("$libexecdir/ksplice-objmanip", $obj_pre, "$obj.KSPLICE_old_code", "keep-old-code");
66 sub do_old_code {
67 my ($out) = @_;
68 my ($obj) = $out =~ /^(.*)\.KSPLICE_old_code$/ or die;
69 my $obj_pre = "$obj.KSPLICE_pre";
70 -e $obj_pre or $obj_pre = $obj;
71 runval("$libexecdir/ksplice-objmanip", $obj_pre, "$obj.KSPLICE_old_code", "keep-old-code");
74 sub link_objs {
75 my ($out, @ins) = @_;
76 if (@ins == 0) {
77 runval(shellwords($ENV{AR}), "rcs", $out);
78 } elsif (@ins == 1) {
79 copy @ins, $out;
80 } else {
81 runval(shellwords($ENV{LD}), "-r", "-o", $out, @ins);
85 sub do_combine {
86 my ($out, @ins) = @_;
87 my @new_code_objs;
88 my @old_code_objs;
89 foreach my $in (@ins) {
90 if (my ($obj) = $in =~ /^(.*)\.KSPLICE$/) {
91 next if (!-s $in);
92 push @new_code_objs, "$obj.KSPLICE_new_code";
93 push @old_code_objs, "$obj.KSPLICE_old_code";
94 } elsif (($obj) = $in =~ /^(.*)\.KSPLICE_old_code$/) {
95 push @old_code_objs, "$obj.KSPLICE_old_code"
96 unless (@old_code_objs && $old_code_objs[$#old_code_objs] eq "$obj.KSPLICE_old_code");
97 } else {
98 die;
102 return empty_diff($out) unless (@old_code_objs);
104 my ($obj) = $out =~ /^(.*)\.KSPLICE$/ or die;
105 link_objs("$obj.KSPLICE_new_code", @new_code_objs);
106 link_objs("$obj.KSPLICE_old_code", @old_code_objs);
108 open OUT, '>', "$out.tmp";
109 print OUT "1\n";
110 close OUT;
111 rename "$out.tmp", $out;
114 sub do_finalize {
115 my ($in, $out, $target) = @_;
116 my $ret = runval_raw("$libexecdir/ksplice-objmanip", $in, $out, "finalize", $target);
117 if ($ret == 0) {
118 } elsif ($ret >> 8 == 66) {
119 runval(shellwords($ENV{AR}), "rcs", $out);
120 } else {
121 child_error();
125 sub do_rmsyms {
126 my ($in, $out) = @_;
127 runval("$libexecdir/ksplice-objmanip", $in, $out, "rmsyms");
130 sub do_system_map_lookup {
131 my ($symarg) = @_;
132 open(SYMS, "<", "$ENV{KSPLICE_CONFIG_DIR}/System.map") or die;
133 my $line;
134 while (defined($line = <SYMS>)) {
135 my ($addr, $type, $sym, $mod) = split(/\s+/, $line);
136 if ($sym eq $symarg) { print $addr; last; }
138 close(SYMS);
141 my %handlers = (
142 'snap' => \&do_snap,
143 'diff' => \&do_diff,
144 'old_code' => \&do_old_code,
145 'combine' => \&do_combine,
146 'finalize' => \&do_finalize,
147 'rmsyms' => \&do_rmsyms,
148 'system_map_lookup' => \&do_system_map_lookup,
151 my ($cmd, @args) = @ARGV;
152 if (exists $handlers{$cmd}) {
153 my $handler = $handlers{$cmd};
154 &$handler(@args);
155 } else {
156 print "Usage: ksplice-obj.pl ", join('|', keys %handlers), " ...\n";
157 exit(1);