1 ############################################################################
3 # This is the top-level Makefile for GHC
8 # Builds GHC, then builds the libraries,
9 # then uses this GHC ("stage 1") to build itself
16 # bootstrap2 + we build GHC one more time ("stage 3")
19 # Just build up to stage 1
22 # Just build stage 2 (stage 1 must be built)
25 # Just build stage 3 (stage 2 must be built)
31 # Install everything, including stage 2 compiler by default
32 # (override with stage=3, for example).
35 # Make a source dist (WARNING: runs 'make distclean' first)
38 # Builds a binary distribution
41 # Builds an HC-file bundle, for bootstrapping
43 # clean, distclean, maintainer-clean
44 # Increasing levels of cleanliness
46 ############################################################################
49 include $(TOP)/mk/boilerplate.mk
52 # Order is important! It's e.g. necessary to descend into include/
53 # before the rest to have a config.h, etc.
55 # If we're booting from .hc files, swap the order
56 # we descend into subdirs - to boot utils must be before driver.
58 .PHONY: stage1 stage2 stage3 bootstrap bootstrap2 bootstrap3
60 # We can't 'make boot' in libraries until stage1 is built
61 ifeq "$(BootingFromHc)" "YES"
62 SUBDIRS_BUILD = gmp includes rts compat compiler docs utils driver
64 SUBDIRS_BUILD = gmp includes compat utils driver docs compiler rts
67 SUBDIRS = gmp includes compat utils driver docs rts libraries compiler libraries/Cabal/doc
69 # Sanity check that all the boot libraries are in the tree, to catch
70 # failure to run darcs-all.
72 @ds=`cat libraries/boot-packages`;\
74 if test ! -d libraries/$$d; then \
75 echo "Looks like you're missing libraries/$$d,"; \
76 echo "maybe you haven't done './darcs-all get'?"; \
80 @if test ! -f libraries/base/configure; then \
81 echo "Looks like you're missing base's configure script."; \
82 echo "Did you run 'sh boot' at the top level?"; \
86 ifeq "$(TARGETPLATFORM)" "i386-unknown-mingw32"
87 ifneq "$(WhatGccIsCalled)" ""
88 GCC_LIB_DEP = stamp.inplace-gcc-lib
92 stage1 : $(GCC_LIB_DEP) check-packages
93 $(MAKE) -C utils/mkdependC boot
94 @case '${MFLAGS}' in *-[ik]*) x_on_err=0;; *-r*[ik]*) x_on_err=0;; *) x_on_err=1;; esac; \
95 for i in $(SUBDIRS_BUILD); do \
96 echo "------------------------------------------------------------------------"; \
97 echo "== $(MAKE) boot $(MFLAGS);"; \
98 echo " in $(shell pwd)/$$i"; \
99 echo "------------------------------------------------------------------------"; \
100 $(MAKE) --no-print-directory -C $$i $(MFLAGS) boot; \
101 if [ $$? -eq 0 -o $$x_on_err -eq 0 ] ; then true; else exit 1; fi; \
103 for i in $(SUBDIRS_BUILD); do \
104 echo "------------------------------------------------------------------------"; \
105 echo "== $(MAKE) all $(MFLAGS);"; \
106 echo " in $(shell pwd)/$$i"; \
107 echo "------------------------------------------------------------------------"; \
108 $(MAKE) --no-print-directory -C $$i $(MFLAGS) all; \
109 if [ $$? -eq 0 -o $$x_on_err -eq 0 ] ; then true; else exit 1; fi; \
111 $(MAKE) -C libraries boot
112 $(MAKE) -C libraries all
114 stage2 : check-packages
115 $(MAKE) -C compiler boot stage=2
116 $(MAKE) -C compiler stage=2
118 stage3 : check-packages
119 $(MAKE) -C compiler boot stage=3
120 $(MAKE) -C compiler stage=3
122 bootstrap : bootstrap2
127 bootstrap3 : bootstrap2
130 ifeq "$(TARGETPLATFORM)" "i386-unknown-mingw32"
131 ifneq "$(WhatGccIsCalled)" ""
132 all :: stamp.inplace-gcc-lib
134 .PHONY: stamp.inplace-gcc-lib
136 # This is a hack to make Cabal able to find ld when we run tests with
137 # the inplace ghc. We should probably install all the gcc stuff in our
138 # tree somewhere, and then have install copy it from there rather than
139 # from the filesystem.
140 stamp.inplace-gcc-lib:
141 $(RM) -r compiler/gcc-lib
142 mkdir compiler/gcc-lib
143 cp $(LD) compiler/gcc-lib
147 $(RM) -r compiler/gcc-lib
148 $(RM) -f inplace-gcc-lib
154 # -----------------------------------------------------------------------------
157 # We want to install the stage 2 bootstrapped compiler by default, but we let
158 # the user override this by saying 'make install stage=1', for example.
160 INSTALL_STAGE = stage=2
165 # Same as default rule, but we pass $(INSTALL_STAGE) to $(MAKE) too
166 install :: check-packages
167 @case '${MFLAGS}' in *-[ik]*) x_on_err=0;; *-r*[ik]*) x_on_err=0;; *) x_on_err=1;; esac; \
168 for i in $(SUBDIRS); do \
169 echo "------------------------------------------------------------------------"; \
170 echo "== $(MAKE) $@ $(MFLAGS);"; \
171 echo " in $(shell pwd)/$$i"; \
172 echo "------------------------------------------------------------------------"; \
173 $(MAKE) --no-print-directory -C $$i $(INSTALL_STAGE) $(MFLAGS) $@; \
174 if [ $$? -eq 0 -o $$x_on_err -eq 0 ] ; then true; else exit 1; fi; \
177 ifeq "$(TARGETPLATFORM)" "i386-unknown-mingw32"
178 # These files need to be in the InstallShield
179 # INSTALL_DATAS rather than INSTALL_DOCS is used so these files go
180 # in the top-level directory of the distribution
181 INSTALL_DATAS += ANNOUNCE LICENSE README
184 # If installing on Windows with MinGW32, copy the gcc compiler, headers and libs
185 # and the perl interpreter and dll into the GHC prefix directory.
186 # Gcc and Perl source locations derived from configuration data.
187 ifeq "$(TARGETPLATFORM)" "i386-unknown-mingw32"
188 ifneq "$(WhatGccIsCalled)" ""
190 -mkdir $(prefix)/gcc-lib
191 -mkdir $(prefix)/include
192 -mkdir $(prefix)/include/mingw
193 -cp -rp $(GccDir)../include/* $(prefix)/include/mingw
194 -cp -rp $(GccDir)../lib/gcc-lib/mingw32/$(GccVersion)/* $(prefix)/gcc-lib
195 -cp -rp $(GccDir)../lib/gcc/mingw32/$(GccVersion)/* $(prefix)/gcc-lib
196 -cp -rp $(GccDir)../libexec/gcc/mingw32/$(GccVersion)/* $(prefix)/gcc-lib
197 -cp $(GccDir)../lib/*.* $(prefix)/gcc-lib
198 -cp $(GccDir)gcc.exe $(prefix)
199 -cp $(GccDir)as.exe $(prefix)/gcc-lib
200 -cp $(GccDir)ld.exe $(prefix)/gcc-lib
201 -cp $(GccDir)dllwrap.exe $(prefix)/gcc-lib
202 -cp $(GccDir)dlltool.exe $(prefix)/gcc-lib
203 -cp $(GhcDir)../perl.exe $(prefix)
204 -cp $(GhcDir)../perl56.dll $(prefix)
208 # Install gcc-extra-opts
210 $(INSTALL_DIR) $(DESTDIR)$(libdir)
211 $(INSTALL_DATA) $(INSTALL_OPTS) extra-gcc-opts $(DESTDIR)$(libdir)
214 @case '${MFLAGS}' in *-[ik]*) x_on_err=0;; *-r*[ik]*) x_on_err=0;; *) x_on_err=1;; esac; \
215 for i in $(SUBDIRS); do \
216 echo "------------------------------------------------------------------------"; \
217 echo "== $(MAKE) $@ $(MFLAGS);"; \
218 echo " in $(shell pwd)/$$i"; \
219 echo "------------------------------------------------------------------------"; \
220 $(MAKE) --no-print-directory -C $$i $(INSTALL_STAGE) $(MFLAGS) $@; \
221 if [ $$? -eq 0 -o $$x_on_err -eq 0 ] ; then true; else exit 1; fi; \
224 # -----------------------------------------------------------------------------
225 # Making a binary distribution
227 # `dist' `binary-dist'
228 # Create a distribution tar file for this program. The tar file
229 # should be set up so that the file names in the tar file start with
230 # a subdirectory name which is the name of the package it is a
231 # distribution for. This name can include the version number.
233 # For example, the distribution tar file of GCC version 1.40 unpacks
234 # into a subdirectory named `gcc-1.40'.
236 # The easiest way to do this is to create a subdirectory
237 # appropriately named, use ln or cp to install the proper files in
238 # it, and then tar that subdirectory.
240 # The dist target should explicitly depend on all non-source files
241 # that are in the distribution, to make sure they are up to date in
242 # the distribution. See Making Releases.
244 # binary-dist is a GHC addition for binary distributions
248 -rm -rf $(BIN_DIST_DIR)
249 -$(RM) $(BIN_DIST_DIR).tar.gz
251 ifeq "$(TARGETPLATFORM)" "i386-unknown-mingw32"
254 $(MAKE) prefix=$(BIN_DIST_DIR) install
257 cd $(BIN_DIST_DIR) && ../distrib/prep-bin-dist-mingw
260 $(MKDIRHIER) $(BIN_DIST_DIR)/icons
261 cp distrib/hsicon.ico $(BIN_DIST_DIR)/icons
265 BinDistDirs = includes compiler docs rts
267 BIN_DIST_TOP= distrib/Makefile \
268 distrib/configure-bin.ac \
279 ifeq "$(darwin_TARGET_OS)" "1"
280 BIN_DIST_TOP+=mk/fix_install_names.sh
283 .PHONY: binary-dist-pre% binary-dist binary-pack
285 binary-dist:: binary-dist-pre
288 $(MKDIRHIER) $(BIN_DIST_DIR)/mk
289 echo 'include $$(TOP)/Makefile-vars' > $(BIN_DIST_DIR)/mk/boilerplate.mk
290 echo 'include $$(TOP)/mk/package.mk' > $(BIN_DIST_DIR)/mk/target.mk
291 echo 'include $$(TOP)/mk/install.mk' >> $(BIN_DIST_DIR)/mk/target.mk
292 echo 'include $$(TOP)/mk/recurse.mk' >> $(BIN_DIST_DIR)/mk/target.mk
293 echo '' > $(BIN_DIST_DIR)/mk/compat.mk
294 cp mk/package.mk $(BIN_DIST_DIR)/mk/
295 cp mk/install.mk $(BIN_DIST_DIR)/mk/
296 cp mk/recurse.mk $(BIN_DIST_DIR)/mk/
297 cp mk/fptools.css $(BIN_DIST_DIR)/mk/
298 $(MKDIRHIER) $(BIN_DIST_DIR)/lib/$(TARGETPLATFORM)
299 $(MKDIRHIER) $(BIN_DIST_DIR)/share
302 $(MAKE) -C gmp binary-dist DOING_BIN_DIST=YES
303 $(MAKE) -C includes binary-dist DOING_BIN_DIST=YES
304 $(MAKE) -C compiler binary-dist DOING_BIN_DIST=YES $(INSTALL_STAGE)
305 $(MAKE) -C rts binary-dist DOING_BIN_DIST=YES
306 $(MAKE) -C driver binary-dist DOING_BIN_DIST=YES
307 $(MAKE) -C utils binary-dist DOING_BIN_DIST=YES
308 $(MAKE) -C docs binary-dist DOING_BIN_DIST=YES
309 $(MAKE) -C libraries binary-dist DOING_BIN_DIST=YES
310 $(MAKE) -C libraries/Cabal/doc binary-dist DOING_BIN_DIST=YES
312 VARFILE=$(BIN_DIST_DIR)/Makefile-vars.in
315 @for i in $(BIN_DIST_TOP); do \
316 if test -f "$$i"; then \
317 echo cp $$i $(BIN_DIST_DIR); \
318 cp $$i $(BIN_DIST_DIR); \
321 @echo "Configuring the Makefile for this project..."
323 echo "package = ghc" >> $(VARFILE)
324 echo "version = $(ProjectVersion)" >> $(VARFILE)
325 echo "ProjectVersion = $(ProjectVersion)" >> $(VARFILE)
326 echo "HaveLibGmp = $(HaveLibGmp)" >> $(VARFILE)
327 echo "GhcLibsWithUnix = $(GhcLibsWithUnix)" >> $(VARFILE)
328 echo "GhcWithInterpreter = $(GhcWithInterpreter)" >> $(VARFILE)
329 echo "GhcHasReadline = $(GhcHasReadline)" >> $(VARFILE)
330 echo "BootingFromHc = $(BootingFromHc)" >> $(VARFILE)
331 echo "XMLDocWays = $(XMLDocWays)" >> $(VARFILE)
332 # We won't actually use xsltproc, but we need to know if it's "" or not
333 echo "XSLTPROC = $(XSLTPROC)" >> $(VARFILE)
334 echo "TARGETPLATFORM = $(TARGETPLATFORM)" >> $(VARFILE)
335 echo "HADDOCK_DOCS = $(HADDOCK_DOCS)" >> $(VARFILE)
337 cat distrib/Makefile-bin-vars.in >> $(VARFILE)
338 @echo "Generating a shippable configure script.."
339 $(MV) $(BIN_DIST_DIR)/configure-bin.ac $(BIN_DIST_DIR)/configure.ac
340 ( cd $(BIN_DIST_DIR); autoreconf )
343 # Tar up the distribution and build a manifest
344 binary-dist :: tar-binary-dist
346 .PHONY: tar-binary-dist
348 ( cd $(BIN_DIST_TOPDIR_ABS); tar cf - $(BIN_DIST_NAME) | bzip2 >$(BIN_DIST_TARBALL) )
349 ( cd $(BIN_DIST_TOPDIR_ABS); bunzip2 -c $(BIN_DIST_TARBALL) | tar tf - | sed "s/^ghc-$(ProjectVersion)/fptools/" | sort >bin-manifest-$(ProjectVersion) )
351 PUBLISH_FILES = $(BIN_DIST_TARBALL)
353 # Upload the distribution and documentation
355 WINDOWS_INSTALLER_BASE = ghc-$(ProjectVersion)-$(TARGETPLATFORM)
356 WINDOWS_INSTALLER = $(WINDOWS_INSTALLER_BASE)$(exeext)
358 PUBLISH_FILES += $(WINDOWS_INSTALLER)
360 binary-dist :: generate-windows-installer
362 .PHONY: generate-windows-installer
363 generate-windows-installer ::
364 $(SED) "s/@VERSION@/$(ProjectVersion)/" distrib/ghc.iss | $(ISCC) /O. /F$(WINDOWS_INSTALLER_BASE) -
367 # Upload the distribution and documentation
368 ifneq "$(PublishLocation)" ""
369 publish :: publish-binary-dist
372 .PHONY: publish-binary-dist
373 publish-binary-dist ::
374 @for f in $(PUBLISH_FILES); do \
375 for i in 0 1 2 3 4 5 6 7 8 9; do \
376 echo "Try $$i: $(PublishCp) $$f $(PublishLocation)/dist"; \
377 if $(PublishCp) $$f $(PublishLocation)/dist; then break; fi; \
381 # You need to first make binddisttest, and then run
382 # make publish 'prefix=$(BIN_DIST_INST_DIR)'
383 # for this to find the right place.
385 # We assume that Windows means Cygwin, as we can't just use docdir
386 # unchanged or rsync (really SSH?) thinks that c:/foo means /foo on
389 ifeq "$(Windows)" "YES"
390 PUBLISH_DOCDIR = $(shell cygpath --unix $(docdir))
392 PUBLISH_DOCDIR = $(docdir)
395 publish-binary-dist ::
396 $(PublishCp) -r $(PUBLISH_DOCDIR)/* $(PublishLocation)/docs
399 @echo "Mechanical and super-natty! Inspect the result and *if* happy; freeze, sell and get some sleep!"
401 # -----------------------------------------------------------------------------
402 # Building source distributions
409 # WARNING: `make dist' calls `make distclean' before tarring up the tree.
415 # Directory in which we're going to build the src dist
417 SRC_DIST_NAME=ghc-$(ProjectVersion)
418 SRC_DIST_DIR=$(shell pwd)/$(SRC_DIST_NAME)
421 # Files to include in source distributions
423 SRC_DIST_DIRS += mk docs distrib bindisttest $(filter-out docs distrib libraries/Cabal/doc,$(SUBDIRS))
425 configure.ac config.guess config.sub configure \
426 aclocal.m4 README ANNOUNCE HACKING LICENSE Makefile install-sh \
427 ghc.spec.in extra-gcc-opts.in VERSION boot
429 # -----------------------------------------------------------------------------
430 # Source distributions
432 # A source dist is built from a complete build tree, because we
433 # require some extra files not contained in a darcs checkout: the
434 # output from Happy and Alex, for example.
436 # The steps performed by 'make dist' are as follows:
437 # - create a complete link-tree of the current build tree in /tmp
438 # - run 'make distclean' on that tree
439 # - remove a bunch of other files that we know shouldn't be in the dist
440 # - tar up first the extralibs package, then the main source package
442 EXTRA_LIBS=$(patsubst %, $(SRC_DIST_NAME)/libraries/%, $(shell cat libraries/extra-packages))
444 SRC_DIST_TARBALL = ghc-$(ProjectVersion)-src.tar.bz2
445 SRC_DIST_EXTRALIBS_TARBALL = ghc-$(ProjectVersion)-src-extralibs.tar.bz2
448 echo $(ProjectVersion) >VERSION
453 $(RM) -rf $(SRC_DIST_DIR)
454 $(RM) $(SRC_DIST_NAME).tar.gz
455 mkdir $(SRC_DIST_DIR)
456 ( cd $(SRC_DIST_DIR) \
457 && for i in $(SRC_DIST_DIRS); do mkdir $$i; (cd $$i && lndir $(FPTOOLS_TOP_ABS)/$$i ); done \
458 && for i in $(SRC_DIST_FILES); do $(LN_S) $(FPTOOLS_TOP_ABS)/$$i .; done \
459 && $(MAKE) distclean \
460 && if test -f $(FPTOOLS_TOP_ABS)/libraries/haskell-src/dist/build/Language/Haskell/Parser.hs; then $(CP) $(FPTOOLS_TOP_ABS)/libraries/haskell-src/dist/build/Language/Haskell/Parser.hs libraries/haskell-src/Language/Haskell/ ; fi \
461 && $(RM) -rf compiler/stage[123] mk/build.mk \
462 && $(FIND) $(SRC_DIST_DIRS) \( -name _darcs -o -name SRC -o -name "autom4te*" -o -name "*~" -o -name ".cvsignore" -o -name "\#*" -o -name ".\#*" -o -name "log" -o -name "*-SAVE" -o -name "*.orig" -o -name "*.rej" \) -print | xargs $(RM) -rf \
464 tar chf - $(EXTRA_LIBS) | bzip2 >$(FPTOOLS_TOP_ABS)/$(SRC_DIST_EXTRALIBS_TARBALL)
465 $(RM) -rf $(EXTRA_LIBS)
466 tar chf - $(SRC_DIST_NAME) 2>$src_log | bzip2 >$(FPTOOLS_TOP_ABS)/$(SRC_DIST_TARBALL)
468 # Upload the distribution(s)
469 # Retrying is to work around buggy firewalls that corrupt large file transfers
471 ifneq "$(PublishLocation)" ""
473 @for i in 0 1 2 3 4 5 6 7 8 9; do \
474 echo "Try $$i: $(PublishCp) $(SRC_DIST_EXTRALIBS_TARBALL) $(PublishLocation)/dist"; \
475 if $(PublishCp) $(SRC_DIST_EXTRALIBS_TARBALL) $(PublishLocation)/dist; then break; fi\
477 @for i in 0 1 2 3 4 5 6 7 8 9; do \
478 echo "Try $$i: $(PublishCp) $(SRC_DIST_TARBALL) $(PublishLocation)/dist"; \
479 if $(PublishCp) $(SRC_DIST_TARBALL) $(PublishLocation)/dist; then break; fi\
483 # -----------------------------------------------------------------------------
487 $(RM) -r ghc-$(ProjectVersion)
488 $(LN_S) . ghc-$(ProjectVersion)
489 $(FIND) ghc-$(ProjectVersion)/compiler \
490 ghc-$(ProjectVersion)/utils \
491 ghc-$(ProjectVersion)/compat \
492 ghc-$(ProjectVersion)/libraries -follow \
493 \( -name "*.hc" -o -name "*_hsc.[ch]" -o -name "*_stub.[ch]" \) -print > hc-files-to-go
494 for f in `$(FIND) ghc-$(ProjectVersion)/compiler ghc-$(ProjectVersion)/utils ghc-$(ProjectVersion)/libraries -name "*.hsc" -follow -print` ""; do \
495 if test "x$$f" != "x" && test -e `echo "$$f" | sed 's/hsc$$/hs/g'`; then \
496 echo `echo "$$f" | sed 's/hsc$$/hs/g' ` >> hc-files-to-go ; \
499 for f in `$(FIND) ghc-$(ProjectVersion)/compiler ghc-$(ProjectVersion)/rts -name "*.cmm" -follow -print` ""; do \
500 if test "x$$f" != "x"; then \
501 echo `echo "$$f" | sed 's/cmm$$/hc/g' ` >> hc-files-to-go ; \
504 echo ghc-$(ProjectVersion)/libraries/base/GHC/PrimopWrappers.hs >> hc-files-to-go
505 echo ghc-$(ProjectVersion)/compiler/parser/Parser.hs >> hc-files-to-go
506 echo ghc-$(ProjectVersion)/compiler/parser/ParserCore.hs >> hc-files-to-go
507 echo ghc-$(ProjectVersion)/compiler/main/ParsePkgConf.hs >> hc-files-to-go
508 echo ghc-$(ProjectVersion)/libraries/haskell-src/Language/Haskell/Parser.hs >> hc-files-to-go
509 tar czf ghc-$(ProjectVersion)-$(TARGETPLATFORM)-hc.tar.gz `cat hc-files-to-go`
511 # -----------------------------------------------------------------------------
514 CLEAN_FILES += hc-files-to-go *-hc.tar.gz
516 DIST_CLEAN_FILES += config.cache config.status mk/config.h mk/stamp-h \
517 ghc.spec docs/users_guide/ug-book.xml extra-gcc-opts
519 # don't clean config.mk: it's needed when cleaning stuff later on
520 LATE_DIST_CLEAN_FILES += mk/config.mk
522 # VERSION is shipped in a source dist
523 MAINTAINER_CLEAN_FILES += VERSION
526 $(RM) -rf autom4te.cache
529 $(MAKE) -C bindisttest $@
530 if test -d testsuite; then $(MAKE) -C testsuite $@; fi
532 # -----------------------------------------------------------------------------
534 # Turn off target.mk's rules for 'all', 'boot' and 'install'.
537 NO_INSTALL_TARGET=YES
539 include $(TOP)/mk/target.mk
541 # -----------------------------------------------------------------------------