diff options
author | Steven Rostedt <srostedt@redhat.com> | 2009-04-29 22:52:21 -0400 |
---|---|---|
committer | Steven Rostedt <rostedt@goodmis.org> | 2009-08-18 21:49:43 -0400 |
commit | dcc60243e726978576cb02b74c84440629c69c87 (patch) | |
tree | f1a4cd4a41c942118a21c3b1cb0b4857fe2c63e1 /scripts/kconfig | |
parent | dc8ed71eeb8adce08d3070f4130e12ee540baa59 (diff) |
kconfig: add streamline_config.pl to scripts
streamline_config.pl is a very powerful tool. For those that install
a kernel to a new box using the config file from the distribution know that
it can take forever to compile the kernel.
Making a custom config file that will still boot your box, but bring
down the compile time of the kernel can be quit painful, and to ask
someone that reported a bug to do this can be a large burdon since that
person may not even know how to build a kernel.
This script will perform "lsmod" to find all the modules loaded on the
current running system. It will read all the Makefiles to map which
CONFIG enables a module. It will read the Kconfig files to find the
dependencies and selects that may be needed to support a CONFIG.
Finally, it reads the .config file and removes any module "=m" that is
not needed to enable the currently loaded modules. The output goes to
standard out.
Here's a way to run the script. From the Linux directory that holds
a distribution .config.
$ scripts/kconfig/streamline_config.pl arch/x86/Kconfig > config-sl
$ mv .config config-save
$ mv config-sl .config
$ make oldconfig
Now you have a .config that will still build all your modules, but also
take much less time to build the kernel.
Signed-off-by: Steven Rostedt <rostedt@goodmis.org>
Diffstat (limited to 'scripts/kconfig')
-rw-r--r-- | scripts/kconfig/streamline_config.pl | 291 |
1 files changed, 291 insertions, 0 deletions
diff --git a/scripts/kconfig/streamline_config.pl b/scripts/kconfig/streamline_config.pl new file mode 100644 index 000000000000..79d85573ee0f --- /dev/null +++ b/scripts/kconfig/streamline_config.pl | |||
@@ -0,0 +1,291 @@ | |||
1 | #!/usr/bin/perl -w | ||
2 | # | ||
3 | # Copywrite 2005-2009 - Steven Rostedt | ||
4 | # Licensed under the terms of the GNU GPL License version 2 | ||
5 | # | ||
6 | # It's simple enough to figure out how this works. | ||
7 | # If not, then you can ask me at stripconfig@goodmis.org | ||
8 | # | ||
9 | # What it does? | ||
10 | # | ||
11 | # If you have installed a Linux kernel from a distribution | ||
12 | # that turns on way too many modules than you need, and | ||
13 | # you only want the modules you use, then this program | ||
14 | # is perfect for you. | ||
15 | # | ||
16 | # It gives you the ability to turn off all the modules that are | ||
17 | # not loaded on your system. | ||
18 | # | ||
19 | # Howto: | ||
20 | # | ||
21 | # 1. Boot up the kernel that you want to stream line the config on. | ||
22 | # 2. Change directory to the directory holding the source of the | ||
23 | # kernel that you just booted. | ||
24 | # 3. Copy the configuraton file to this directory as .config | ||
25 | # 4. Have all your devices that you need modules for connected and | ||
26 | # operational (make sure that their corresponding modules are loaded) | ||
27 | # 5. Run this script redirecting the output to some other file | ||
28 | # like config_strip. | ||
29 | # 6. Back up your old config (if you want too). | ||
30 | # 7. copy the config_strip file to .config | ||
31 | # 8. Run "make oldconfig" | ||
32 | # | ||
33 | # Now your kernel is ready to be built with only the modules that | ||
34 | # are loaded. | ||
35 | # | ||
36 | # Here's what I did with my Debian distribution. | ||
37 | # | ||
38 | # cd /usr/src/linux-2.6.10 | ||
39 | # cp /boot/config-2.6.10-1-686-smp .config | ||
40 | # ~/bin/streamline_config > config_strip | ||
41 | # mv .config config_sav | ||
42 | # mv config_strip .config | ||
43 | # make oldconfig | ||
44 | # | ||
45 | my $config = ".config"; | ||
46 | my $linuxpath = "."; | ||
47 | |||
48 | open(CIN,$config) || die "Can't open current config file: $config"; | ||
49 | my @makefiles = `find $linuxpath -name Makefile`; | ||
50 | my %depends; | ||
51 | my %selects; | ||
52 | my %prompts; | ||
53 | my %objects; | ||
54 | my $var; | ||
55 | my $cont = 0; | ||
56 | |||
57 | # Get the top level Kconfig file (passed in) | ||
58 | my $kconfig = $ARGV[0]; | ||
59 | |||
60 | # prevent recursion | ||
61 | my %read_kconfigs; | ||
62 | |||
63 | sub read_kconfig { | ||
64 | my ($kconfig) = @_; | ||
65 | |||
66 | my $state = "NONE"; | ||
67 | my $config; | ||
68 | my @kconfigs; | ||
69 | |||
70 | open(KIN, $kconfig) || die "Can't open $kconfig"; | ||
71 | while (<KIN>) { | ||
72 | chomp; | ||
73 | |||
74 | # collect any Kconfig sources | ||
75 | if (/^source\s*"(.*)"/) { | ||
76 | $kconfigs[$#kconfigs+1] = $1; | ||
77 | } | ||
78 | |||
79 | # configs found | ||
80 | if (/^\s*config\s+(\S+)\s*$/) { | ||
81 | $state = "NEW"; | ||
82 | $config = $1; | ||
83 | |||
84 | # collect the depends for the config | ||
85 | } elsif ($state eq "NEW" && /^\s*depends\s+on\s+(.*)$/) { | ||
86 | $state = "DEP"; | ||
87 | $depends{$config} = $1; | ||
88 | } elsif ($state eq "DEP" && /^\s*depends\s+on\s+(.*)$/) { | ||
89 | $depends{$config} .= " " . $1; | ||
90 | |||
91 | # Get the configs that select this config | ||
92 | } elsif ($state ne "NONE" && /^\s*select\s+(\S+)/) { | ||
93 | if (defined($selects{$1})) { | ||
94 | $selects{$1} .= " " . $config; | ||
95 | } else { | ||
96 | $selects{$1} = $config; | ||
97 | } | ||
98 | |||
99 | # configs without prompts must be selected | ||
100 | } elsif ($state ne "NONE" && /^\s*tristate\s\S/) { | ||
101 | # note if the config has a prompt | ||
102 | $prompt{$config} = 1; | ||
103 | |||
104 | # stop on "help" | ||
105 | } elsif (/^\s*help\s*$/) { | ||
106 | $state = "NONE"; | ||
107 | } | ||
108 | } | ||
109 | close(KIN); | ||
110 | |||
111 | # read in any configs that were found. | ||
112 | foreach $kconfig (@kconfigs) { | ||
113 | if (!defined($read_kconfigs{$kconfig})) { | ||
114 | $read_kconfigs{$kconfig} = 1; | ||
115 | read_kconfig($kconfig); | ||
116 | } | ||
117 | } | ||
118 | } | ||
119 | |||
120 | if ($kconfig) { | ||
121 | read_kconfig($kconfig); | ||
122 | } | ||
123 | |||
124 | # Read all Makefiles to map the configs to the objects | ||
125 | foreach my $makefile (@makefiles) { | ||
126 | chomp $makefile; | ||
127 | |||
128 | open(MIN,$makefile) || die "Can't open $makefile"; | ||
129 | while (<MIN>) { | ||
130 | my $objs; | ||
131 | |||
132 | # is this a line after a line with a backslash? | ||
133 | if ($cont && /(\S.*)$/) { | ||
134 | $objs = $1; | ||
135 | } | ||
136 | $cont = 0; | ||
137 | |||
138 | # collect objects after obj-$(CONFIG_FOO_BAR) | ||
139 | if (/obj-\$\((CONFIG_[^\)]*)\)\s*[+:]?=\s*(.*)/) { | ||
140 | $var = $1; | ||
141 | $objs = $2; | ||
142 | } | ||
143 | if (defined($objs)) { | ||
144 | # test if the line ends with a backslash | ||
145 | if ($objs =~ m,(.*)\\$,) { | ||
146 | $objs = $1; | ||
147 | $cont = 1; | ||
148 | } | ||
149 | |||
150 | foreach my $obj (split /\s+/,$objs) { | ||
151 | $obj =~ s/-/_/g; | ||
152 | if ($obj =~ /(.*)\.o$/) { | ||
153 | # Objects may bes enabled by more than one config. | ||
154 | # Store configs in an array. | ||
155 | my @arr; | ||
156 | |||
157 | if (defined($objects{$1})) { | ||
158 | @arr = @{$objects{$1}}; | ||
159 | } | ||
160 | |||
161 | $arr[$#arr+1] = $var; | ||
162 | |||
163 | # The objects have a hash mapping to a reference | ||
164 | # of an array of configs. | ||
165 | $objects{$1} = \@arr; | ||
166 | } | ||
167 | } | ||
168 | } | ||
169 | } | ||
170 | close(MIN); | ||
171 | } | ||
172 | |||
173 | my %modules; | ||
174 | |||
175 | # see what modules are loaded on this system | ||
176 | open(LIN,"/sbin/lsmod|") || die "Cant lsmod"; | ||
177 | while (<LIN>) { | ||
178 | next if (/^Module/); # Skip the first line. | ||
179 | if (/^(\S+)/) { | ||
180 | $modules{$1} = 1; | ||
181 | } | ||
182 | } | ||
183 | close (LIN); | ||
184 | |||
185 | # add to the configs hash all configs that are needed to enable | ||
186 | # a loaded module. | ||
187 | my %configs; | ||
188 | foreach my $module (keys(%modules)) { | ||
189 | if (defined($objects{$module})) { | ||
190 | @arr = @{$objects{$module}}; | ||
191 | foreach my $conf (@arr) { | ||
192 | $configs{$conf} = $module; | ||
193 | } | ||
194 | } else { | ||
195 | # Most likely, someone has a custom (binary?) module loaded. | ||
196 | print STDERR "$module config not found!!\n"; | ||
197 | } | ||
198 | } | ||
199 | |||
200 | my $valid = "A-Za-z_0-9"; | ||
201 | my $repeat = 1; | ||
202 | |||
203 | # | ||
204 | # Note, we do not care about operands (like: &&, ||, !) we want to add any | ||
205 | # config that is in the depend list of another config. This script does | ||
206 | # not enable configs that are not already enabled. If we come across a | ||
207 | # config A that depends on !B, we can still add B to the list of depends | ||
208 | # to keep on. If A was on in the original config, B would not have been | ||
209 | # and B would not be turned on by this script. | ||
210 | # | ||
211 | sub parse_config_dep_select | ||
212 | { | ||
213 | my ($p) = @_; | ||
214 | |||
215 | while ($p =~ /[$valid]/) { | ||
216 | |||
217 | if ($p =~ /^[^$valid]*([$valid]+)/) { | ||
218 | my $conf = "CONFIG_" . $1; | ||
219 | |||
220 | $p =~ s/^[^$valid]*[$valid]+//; | ||
221 | |||
222 | if (!defined($configs{$conf})) { | ||
223 | # We must make sure that this config has its | ||
224 | # dependencies met. | ||
225 | $repeat = 1; # do again | ||
226 | $configs{$conf} = 1; | ||
227 | } | ||
228 | } else { | ||
229 | die "this should never happen"; | ||
230 | } | ||
231 | } | ||
232 | } | ||
233 | |||
234 | while ($repeat) { | ||
235 | $repeat = 0; | ||
236 | |||
237 | foreach my $config (keys %configs) { | ||
238 | $config =~ s/^CONFIG_//; | ||
239 | |||
240 | if (!defined($depends{$config})) { | ||
241 | next; | ||
242 | } | ||
243 | |||
244 | # This config has dependencies. Make sure they are also included | ||
245 | parse_config_dep_select $depends{$config}; | ||
246 | |||
247 | if (defined($prompt{$config}) || !defined($selects{$config})) { | ||
248 | next; | ||
249 | } | ||
250 | |||
251 | # config has no prompt and must be selected. | ||
252 | parse_config_dep_select $selects{$config}; | ||
253 | } | ||
254 | } | ||
255 | |||
256 | my %setconfigs; | ||
257 | |||
258 | # Finally, read the .config file and turn off any module enabled that | ||
259 | # we could not find a reason to keep enabled. | ||
260 | while(<CIN>) { | ||
261 | if (/^(CONFIG.*)=m/) { | ||
262 | if (defined($configs{$1})) { | ||
263 | $setconfigs{$1} = 1; | ||
264 | print; | ||
265 | } else { | ||
266 | print "# $1 is not set\n"; | ||
267 | } | ||
268 | } else { | ||
269 | print; | ||
270 | } | ||
271 | } | ||
272 | close(CIN); | ||
273 | |||
274 | # Integrity check, make sure all modules that we want enabled do | ||
275 | # indeed have their configs set. | ||
276 | loop: | ||
277 | foreach my $module (keys(%modules)) { | ||
278 | if (defined($objects{$module})) { | ||
279 | my @arr = @{$objects{$module}}; | ||
280 | foreach my $conf (@arr) { | ||
281 | if (defined($setconfigs{$conf})) { | ||
282 | next loop; | ||
283 | } | ||
284 | } | ||
285 | print STDERR "module $module did not have configs"; | ||
286 | foreach my $conf (@arr) { | ||
287 | print STDERR " " , $conf; | ||
288 | } | ||
289 | print STDERR "\n"; | ||
290 | } | ||
291 | } | ||