2010-05-18 Justus Winter <4winter@informatik.uni-hamburg.de>
[grub.git] / genmk.rb
blobe62dbd4f6d502ce9dda756e583f7d5a685cc5688
1 #! /usr/bin/ruby -w
3 # Copyright (C) 2002,2003,2004,2005,2006,2007,2008,2009  Free Software Foundation, Inc.
5 # This genmk.rb is free software; the author
6 # gives unlimited permission to copy and/or distribute it,
7 # with or without modifications, as long as this notice is preserved.
9 # This program is distributed in the hope that it will be useful,
10 # but WITHOUT ANY WARRANTY, to the extent permitted by law; without
11 # even the implied warranty of MERCHANTABILITY or FITNESS FOR A
12 # PARTICULAR PURPOSE.
14 module Enumerable
15   def collect_with_index
16     ret = []
17     self.each_with_index do |item, index|
18       ret.push(yield(item, index))
19     end
20     ret
21   end
22 end
24 class String
25   def to_var
26     self.gsub(/[^a-zA-Z0-9_@]/, '_')
27   end
29   def suffix(str)
30     self.sub(/\.[^\.]*$/, '') + '.' + str
31   end
33   def to_obj
34     self.sub(/\.[^\.]*$/, '').to_var + '.o'
35   end
36 end
38 class Image
39   def initialize(dir, name)
40     @dir = dir
41     @name = name
42     @rule_count = 0
43   end
44   attr_reader :dir, :name
46   def rule(sources)
47     prefix = @name.to_var
48     @rule_count += 1
49     exe = @name.suffix('exec')
50     objs = sources.collect do |src|
51       raise "unknown source file `#{src}'" if /\.[cS]$/ !~ src
52       prefix + '-' + src.to_obj
53     end
54     objs_str = objs.join(' ')
55     deps = objs.collect {|obj| obj.suffix('d')}
56     deps_str = deps.join(' ')
59 clean-image-#{@name}.#{@rule_count}:
60         rm -f #{@name} #{exe} #{objs_str}
62 CLEAN_IMAGE_TARGETS += clean-image-#{@name}.#{@rule_count}
64 mostlyclean-image-#{@name}.#{@rule_count}:
65         rm -f #{deps_str}
67 MOSTLYCLEAN_IMAGE_TARGETS += mostlyclean-image-#{@name}.#{@rule_count}
69 ifneq ($(TARGET_APPLE_CC),1)
70 #{@name}: #{exe}
71         $(OBJCOPY) -O $(#{prefix}_FORMAT) --strip-unneeded -R .note -R .comment -R .note.gnu.build-id -R .reginfo -R .rel.dyn $< $@
72 else
73 ifneq (#{exe},kernel.exec)
74 #{@name}: #{exe} ./grub-macho2img
75         ./grub-macho2img $< $@
76 else
77 #{@name}: #{exe} ./grub-macho2img
78         ./grub-macho2img --bss $< $@
79 endif
80 endif
82 #{exe}: #{objs_str}
83         $(TARGET_CC) -o $@ $^ $(TARGET_LDFLAGS) $(#{prefix}_LDFLAGS)
85 " + objs.collect_with_index do |obj, i|
86       src = sources[i]
87       fake_obj = File.basename(src).suffix('o')
88       dep = deps[i]
89       flag = if /\.c$/ =~ src then 'CFLAGS' else 'ASFLAGS' end
90       extra_flags = if /\.S$/ =~ src then '-DASM_FILE=1' else '' end
91       dir = File.dirname(src)
93       "#{obj}: #{src} $(#{src}_DEPENDENCIES)
94         $(TARGET_CC) -I#{dir} -I$(srcdir)/#{dir} $(TARGET_CPPFLAGS) #{extra_flags} $(TARGET_#{flag}) $(#{prefix}_#{flag}) -DGRUB_FILE=\\\"#{src}\\\" -MD -c -o $@ $<
95 -include #{dep}
98     end.join('')
99   end
102 # Use PModule instead Module, to avoid name conflicting.
103 class PModule
104   def initialize(dir, name)
105     @dir = dir
106     @name = name
107     @rule_count = 0
108   end
109   attr_reader :dir, :name
111   def rule(sources)
112     prefix = @name.to_var
113     @rule_count += 1
114     objs = sources.collect do |src|
115       raise "unknown source file `#{src}'" if /\.[cS]$/ !~ src
116       prefix + '-' + src.to_obj
117     end
118     objs_str = objs.join(' ')
119     deps = objs.collect {|obj| obj.suffix('d')}
120     deps_str = deps.join(' ')
121     pre_obj = 'pre-' + @name.suffix('o')
122     mod_src = 'mod-' + @name.suffix('c')
123     mod_obj = mod_src.suffix('o')
124     defsym = 'def-' + @name.suffix('lst')
125     undsym = 'und-' + @name.suffix('lst')
126     mod_name = File.basename(@name, '.mod')
127     symbolic_name = mod_name.sub(/\.[^\.]*$/, '')
130 clean-module-#{@name}.#{@rule_count}:
131         rm -f #{@name} #{mod_obj} #{mod_src} #{pre_obj} #{objs_str} #{undsym}
133 CLEAN_MODULE_TARGETS += clean-module-#{@name}.#{@rule_count}
135 clean-module-#{@name}-symbol.#{@rule_count}:
136         rm -f #{defsym}
138 CLEAN_MODULE_TARGETS += clean-module-#{@name}-symbol.#{@rule_count}
139 DEFSYMFILES += #{defsym}
140 mostlyclean-module-#{@name}.#{@rule_count}:
141         rm -f #{deps_str}
143 MOSTLYCLEAN_MODULE_TARGETS += mostlyclean-module-#{@name}.#{@rule_count}
144 UNDSYMFILES += #{undsym}
146 ifeq ($(TARGET_NO_MODULES), yes)
147 #{@name}: #{pre_obj} $(TARGET_OBJ2ELF)
148         -rm -f $@
149         $(TARGET_CC) $(#{prefix}_LDFLAGS) $(TARGET_LDFLAGS) -Wl,-r,-d -o $@ #{pre_obj}
150         if test ! -z \"$(TARGET_OBJ2ELF)\"; then ./$(TARGET_OBJ2ELF) $@ || (rm -f $@; exit 1); fi
151         if test x$(TARGET_NO_STRIP) != xyes ; then $(STRIP) --strip-unneeded -K grub_mod_init -K grub_mod_fini -K _grub_mod_init -K _grub_mod_fini -R .note -R .comment $@; fi
152 else
153 ifneq ($(TARGET_APPLE_CC),1)
154 #{@name}: #{pre_obj} #{mod_obj} $(TARGET_OBJ2ELF)
155         -rm -f $@
156         $(TARGET_CC) $(#{prefix}_LDFLAGS) $(TARGET_LDFLAGS) -Wl,-r,-d -o $@ #{pre_obj} #{mod_obj}
157         if test ! -z \"$(TARGET_OBJ2ELF)\"; then ./$(TARGET_OBJ2ELF) $@ || (rm -f $@; exit 1); fi
158         if test x$(TARGET_NO_STRIP) != xyes ; then $(STRIP) --strip-unneeded -K grub_mod_init -K grub_mod_fini -K _grub_mod_init -K _grub_mod_fini -R .note -R .comment $@; fi
159 else
160 #{@name}: #{pre_obj} #{mod_obj} $(TARGET_OBJ2ELF)
161         -rm -f $@
162         -rm -f $@.bin
163         $(TARGET_CC) $(#{prefix}_LDFLAGS) $(TARGET_LDFLAGS) -Wl,-r,-d -o $@.bin #{pre_obj} #{mod_obj}
164         $(OBJCONV) -f$(TARGET_MODULE_FORMAT) -nr:_grub_mod_init:grub_mod_init -nr:_grub_mod_fini:grub_mod_fini -wd1106 -ew2030 -ew2050 -nu -nd $@.bin $@
165         -rm -f $@.bin
166 endif
167 endif
169 #{pre_obj}: $(#{prefix}_DEPENDENCIES) #{objs_str}
170         -rm -f $@
171         $(TARGET_CC) $(#{prefix}_LDFLAGS) $(TARGET_LDFLAGS) -Wl,-r,-d -o $@ #{objs_str}
173 #{mod_obj}: #{mod_src}
174         $(TARGET_CC) $(TARGET_CPPFLAGS) $(TARGET_CFLAGS) $(#{prefix}_CFLAGS) -DGRUB_FILE=\\\"#{mod_src}\\\" -c -o $@ $<
176 #{mod_src}: $(builddir)/moddep.lst $(srcdir)/genmodsrc.sh
177         sh $(srcdir)/genmodsrc.sh '#{mod_name}' $< > $@ || (rm -f $@; exit 1)
179 ifneq ($(TARGET_APPLE_CC),1)
180 #{defsym}: #{pre_obj}
181         $(NM) -g --defined-only -P -p $< | sed 's/^\\([^ ]*\\).*/\\1 #{mod_name}/' > $@
182 else
183 #{defsym}: #{pre_obj}
184         $(NM) -g -P -p $< | grep -E '^[a-zA-Z0-9_]* [TDS]'  | sed 's/^\\([^ ]*\\).*/\\1 #{mod_name}/' > $@
185 endif
187 #{undsym}: #{pre_obj}
188         echo '#{mod_name}' > $@
189         $(NM) -u -P -p $< | cut -f1 -d' ' >> $@
191 " + objs.collect_with_index do |obj, i|
192       src = sources[i]
193       fake_obj = File.basename(src).suffix('o')
194       extra_target = obj.sub(/\.[^\.]*$/, '') + '-extra'
195       command = 'cmd-' + obj.suffix('lst')
196       fs = 'fs-' + obj.suffix('lst')
197       partmap = 'partmap-' + obj.suffix('lst')
198       handler = 'handler-' + obj.suffix('lst')
199       terminal = 'terminal-' + obj.suffix('lst')
200       parttool = 'parttool-' + obj.suffix('lst')
201       video = 'video-' + obj.suffix('lst')
202       dep = deps[i]
203       flag = if /\.c$/ =~ src then 'CFLAGS' else 'ASFLAGS' end
204       extra_flags = if /\.S$/ =~ src then '-DASM_FILE=1' else '' end
205       dir = File.dirname(src)
207       "#{obj}: #{src} $(#{src}_DEPENDENCIES)
208         $(TARGET_CC) -I#{dir} -I$(srcdir)/#{dir} $(TARGET_CPPFLAGS) #{extra_flags} $(TARGET_#{flag}) $(#{prefix}_#{flag}) -DGRUB_FILE=\\\"#{src}\\\" -MD -c -o $@ $<
209 -include #{dep}
211 clean-module-#{extra_target}.#{@rule_count}:
212         rm -f #{command} #{fs} #{partmap} #{handler} #{parttool} #{video} #{terminal}
214 CLEAN_MODULE_TARGETS += clean-module-#{extra_target}.#{@rule_count}
216 COMMANDFILES += #{command}
217 FSFILES += #{fs}
218 PARTTOOLFILES += #{parttool}
219 PARTMAPFILES += #{partmap}
220 HANDLERFILES += #{handler}
221 TERMINALFILES += #{terminal}
222 VIDEOFILES += #{video}
224 #{command}: #{src} $(#{src}_DEPENDENCIES) gencmdlist.sh
225         set -e; \
226           $(TARGET_CC) -I#{dir} -I$(srcdir)/#{dir} $(TARGET_CPPFLAGS) #{extra_flags} $(TARGET_#{flag}) $(#{prefix}_#{flag}) -E $< \
227           | sh $(srcdir)/gencmdlist.sh #{symbolic_name} > $@ || (rm -f $@; exit 1)
229 #{fs}: #{src} $(#{src}_DEPENDENCIES) genfslist.sh
230         set -e; \
231           $(TARGET_CC) -I#{dir} -I$(srcdir)/#{dir} $(TARGET_CPPFLAGS) #{extra_flags} $(TARGET_#{flag}) $(#{prefix}_#{flag}) -E $< \
232           | sh $(srcdir)/genfslist.sh #{symbolic_name} > $@ || (rm -f $@; exit 1)
234 #{parttool}: #{src} $(#{src}_DEPENDENCIES) genparttoollist.sh
235         set -e; \
236           $(TARGET_CC) -I#{dir} -I$(srcdir)/#{dir} $(TARGET_CPPFLAGS) #{extra_flags} $(TARGET_#{flag}) $(#{prefix}_#{flag}) -E $< \
237           | sh $(srcdir)/genparttoollist.sh #{symbolic_name} > $@ || (rm -f $@; exit 1)
239 #{partmap}: #{src} $(#{src}_DEPENDENCIES) genpartmaplist.sh
240         set -e; \
241           $(TARGET_CC) -I#{dir} -I$(srcdir)/#{dir} $(TARGET_CPPFLAGS) #{extra_flags} $(TARGET_#{flag}) $(#{prefix}_#{flag}) -E $< \
242           | sh $(srcdir)/genpartmaplist.sh #{symbolic_name} > $@ || (rm -f $@; exit 1)
244 #{handler}: #{src} $(#{src}_DEPENDENCIES) genhandlerlist.sh
245         set -e; \
246           $(TARGET_CC) -I#{dir} -I$(srcdir)/#{dir} $(TARGET_CPPFLAGS) #{extra_flags} $(TARGET_#{flag}) $(#{prefix}_#{flag}) -E $< \
247           | sh $(srcdir)/genhandlerlist.sh #{symbolic_name} > $@ || (rm -f $@; exit 1)
249 #{terminal}: #{src} $(#{src}_DEPENDENCIES) genterminallist.sh
250         set -e; \
251           $(TARGET_CC) -I#{dir} -I$(srcdir)/#{dir} $(TARGET_CPPFLAGS) #{extra_flags} $(TARGET_#{flag}) $(#{prefix}_#{flag}) -E $< \
252           | sh $(srcdir)/genterminallist.sh #{symbolic_name} > $@ || (rm -f $@; exit 1)
254 #{video}: #{src} $(#{src}_DEPENDENCIES) genvideolist.sh
255         set -e; \
256           $(TARGET_CC) -I#{dir} -I$(srcdir)/#{dir} $(TARGET_CPPFLAGS) #{extra_flags} $(TARGET_#{flag}) $(#{prefix}_#{flag}) -E $< \
257           | sh $(srcdir)/genvideolist.sh #{symbolic_name} > $@ || (rm -f $@; exit 1)
260     end.join('')
261   end
264 class Utility
265   def initialize(dir, name)
266     @dir = dir
267     @name = name
268     @rule_count = 0
269   end
270   def print_tail()
271     prefix = @name.to_var
272     print "#{@name}: $(#{prefix}_DEPENDENCIES) $(#{prefix}_OBJECTS)
273         $(CC) -o $@ $(#{prefix}_OBJECTS) $(LDFLAGS) $(#{prefix}_LDFLAGS)
276   end
277   attr_reader :dir, :name
279   def rule(sources)
280     prefix = @name.to_var
281     @rule_count += 1
282     objs = sources.collect do |src|
283       raise "unknown source file `#{src}'" if /\.[cS]$/ !~ src
284       prefix + '-' + src.to_obj
285     end
286     objs_str = objs.join(' ');
287     deps = objs.collect {|obj| obj.suffix('d')}
288     deps_str = deps.join(' ');
290     "
291 clean-utility-#{@name}.#{@rule_count}:
292         rm -f #{@name}$(EXEEXT) #{objs_str}
294 CLEAN_UTILITY_TARGETS += clean-utility-#{@name}.#{@rule_count}
296 mostlyclean-utility-#{@name}.#{@rule_count}:
297         rm -f #{deps_str}
299 MOSTLYCLEAN_UTILITY_TARGETS += mostlyclean-utility-#{@name}.#{@rule_count}
301 #{prefix}_OBJECTS += #{objs_str}
303 " + objs.collect_with_index do |obj, i|
304       src = sources[i]
305       fake_obj = File.basename(src).suffix('o')
306       dep = deps[i]
307       dir = File.dirname(src)
309       "#{obj}: #{src} $(#{src}_DEPENDENCIES)
310         $(CC) -I#{dir} -I$(srcdir)/#{dir} $(CPPFLAGS) $(CFLAGS) -DGRUB_UTIL=1 $(#{prefix}_CFLAGS) -DGRUB_FILE=\\\"#{src}\\\" -MD -c -o $@ $<
311 -include #{dep}
314     end.join('')
315   end
318 class Program
319   def initialize(dir, name)
320     @dir = dir
321     @name = name
322   end
323   attr_reader :dir, :name
325   def print_tail()
326     prefix = @name.to_var
327     print "CLEANFILES += #{@name} $(#{prefix}_OBJECTS)
328 ifeq ($(#{prefix}_RELOCATABLE),yes)
329 #{@name}: $(#{prefix}_DEPENDENCIES) $(#{prefix}_OBJECTS)
330         $(TARGET_CC) -Wl,-r,-d -o $@ $(#{prefix}_OBJECTS) $(TARGET_LDFLAGS) $(#{prefix}_LDFLAGS)
331         if test x$(TARGET_NO_STRIP) != xyes ; then $(STRIP) --strip-unneeded -K start -R .note -R .comment $@; fi
332 else
333 #{@name}: $(#{prefix}_DEPENDENCIES) $(#{prefix}_OBJECTS)
334         $(TARGET_CC) -o $@ $(#{prefix}_OBJECTS) $(TARGET_LDFLAGS) $(#{prefix}_LDFLAGS)
335         if test x$(TARGET_NO_STRIP) != xyes ; then $(STRIP) -R .rel.dyn -R .reginfo -R .note -R .comment $@; fi
336 endif
339   end
341   def rule(sources)
342     prefix = @name.to_var
343     objs = sources.collect do |src|
344       raise "unknown source file `#{src}'" if /\.[cS]$/ !~ src
345       prefix + '-' + src.to_obj
346     end
347     deps = objs.collect {|obj| obj.suffix('d')}
348     deps_str = deps.join(' ');
350     "MOSTLYCLEANFILES += #{deps_str}
352 " + objs.collect_with_index do |obj, i|
353       src = sources[i]
354       fake_obj = File.basename(src).suffix('o')
355       dep = deps[i]
356       flag = if /\.c$/ =~ src then 'CFLAGS' else 'ASFLAGS' end
357       extra_flags = if /\.S$/ =~ src then '-DASM_FILE=1' else '' end
358       dir = File.dirname(src)
360       "#{obj}: #{src} $(#{src}_DEPENDENCIES)
361         $(TARGET_CC) -I#{dir} -I$(srcdir)/#{dir} $(TARGET_CPPFLAGS) #{extra_flags} $(TARGET_#{flag}) $(#{prefix}_#{flag}) -DGRUB_FILE=\\\"#{src}\\\" -MD -c -o $@ $<
363 -include #{dep}
365 #{prefix}_OBJECTS += #{obj}
367     end.join('')
368   end
371 class Script
372   def initialize(dir, name)
373     @dir = dir
374     @name = name
375   end
376   attr_reader :dir, :name
378   def rule(sources)
379     if sources.length != 1
380       raise "only a single source file must be specified for a script"
381     end
382     src = sources[0]
383     if /\.in$/ !~ src
384       raise "unknown source file `#{src}'"
385     end
387     "CLEANFILES += #{@name}
389 #{@name}: #{src} $(#{src}_DEPENDENCIES) config.status
390         ./config.status --file=-:#{src} | sed -e 's,@pkglib_DATA@,$(pkglib_DATA),g' > $@
391         chmod +x $@
394   end
397 images = []
398 utils = []
399 pmodules = []
400 programs = []
401 scripts = []
403 l = gets
404 print l
405 print "# Generated by genmk.rb, please don't edit!\n"
407 cont = false
408 str = nil
409 while l = gets
410   if cont
411     str += l
412   else
413     str = l
414   end
416   print l
417   cont = (/\\$/ =~ l)
418   unless cont
419     str.gsub!(/\\\n/, ' ')
421     if /^([a-zA-Z0-9_]+)\s*\+?=\s*(.*?)\s*$/ =~ str
422       var, args = $1, $2
424       if var =~ /^([a-zA-Z0-9_]+)_([A-Z]+)$/
425         prefix, type = $1, $2
427         case type
428         when 'IMAGES'
429           images += args.split(/\s+/).collect do |img|
430             Image.new(prefix, img)
431           end
433         when 'MODULES'
434           pmodules += args.split(/\s+/).collect do |pmod|
435             PModule.new(prefix, pmod)
436           end
438         when 'UTILITIES'
439           utils += args.split(/\s+/).collect do |util|
440             Utility.new(prefix, util)
441           end
443         when 'PROGRAMS'
444           programs += args.split(/\s+/).collect do |prog|
445             Program.new(prefix, prog)
446           end
448         when 'SCRIPTS'
449           scripts += args.split(/\s+/).collect do |script|
450             Script.new(prefix, script)
451           end
453         when 'SOURCES'
454           if img = images.detect() {|i| i.name.to_var == prefix}
455             print img.rule(args.split(/\s+/))
456           elsif pmod = pmodules.detect() {|m| m.name.to_var == prefix}
457             print pmod.rule(args.split(/\s+/))
458           elsif util = utils.detect() {|u| u.name.to_var == prefix}
459             print util.rule(args.split(/\s+/))
460           elsif program = programs.detect() {|u| u.name.to_var == prefix}
461             print program.rule(args.split(/\s+/))
462           elsif script = scripts.detect() {|s| s.name.to_var == prefix}
463             print script.rule(args.split(/\s+/))
464           end
465         end
466       end
468     end
470   end
473 utils.each {|util| util.print_tail()}
474 programs.each {|program| program.print_tail()}