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 libraries/Cabal/doc
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
166 $(MAKE) INSTALL_PROGRAM='$(INSTALL_PROGRAM) -s' install
168 # Same as default rule, but we pass $(INSTALL_STAGE) to $(MAKE) too
169 install :: check-packages
170 @case '${MFLAGS}' in *-[ik]*) x_on_err=0;; *-r*[ik]*) x_on_err=0;; *) x_on_err=1;; esac; \
171 for i in $(SUBDIRS); do \
172 echo "------------------------------------------------------------------------"; \
173 echo "== $(MAKE) $@ $(MFLAGS);"; \
174 echo " in $(shell pwd)/$$i"; \
175 echo "------------------------------------------------------------------------"; \
176 $(MAKE) --no-print-directory -C $$i $(INSTALL_STAGE) $(MFLAGS) $@; \
177 if [ $$? -eq 0 -o $$x_on_err -eq 0 ] ; then true; else exit 1; fi; \
180 ifeq "$(TARGETPLATFORM)" "i386-unknown-mingw32"
181 # These files need to be in the InstallShield
182 # INSTALL_DATAS rather than INSTALL_DOCS is used so these files go
183 # in the top-level directory of the distribution
184 INSTALL_DATAS += ANNOUNCE LICENSE README
187 # If installing on Windows with MinGW32, copy the gcc compiler, headers and libs
188 # and the perl interpreter and dll into the GHC prefix directory.
189 # Gcc and Perl source locations derived from configuration data.
190 ifeq "$(TARGETPLATFORM)" "i386-unknown-mingw32"
191 ifneq "$(WhatGccIsCalled)" ""
193 -mkdir $(prefix)/gcc-lib
194 -mkdir $(prefix)/include
195 -mkdir $(prefix)/include/mingw
196 -cp -rp $(GccDir)../include/* $(prefix)/include/mingw
197 -cp -rp $(GccDir)../lib/gcc-lib/mingw32/$(GccVersion)/* $(prefix)/gcc-lib
198 -cp -rp $(GccDir)../lib/gcc/mingw32/$(GccVersion)/* $(prefix)/gcc-lib
199 -cp -rp $(GccDir)../libexec/gcc/mingw32/$(GccVersion)/* $(prefix)/gcc-lib
200 -cp $(GccDir)../lib/*.* $(prefix)/gcc-lib
201 -cp $(GccDir)gcc.exe $(prefix)
202 -cp $(GccDir)as.exe $(prefix)/gcc-lib
203 -cp $(GccDir)ld.exe $(prefix)/gcc-lib
204 -cp $(GccDir)dllwrap.exe $(prefix)/gcc-lib
205 -cp $(GccDir)dlltool.exe $(prefix)/gcc-lib
206 -cp $(GhcDir)../perl.exe $(prefix)
207 -cp $(GhcDir)../perl56.dll $(prefix)
211 # Install gcc-extra-opts
213 $(INSTALL_DIR) $(DESTDIR)$(libdir)
214 $(INSTALL_DATA) $(INSTALL_OPTS) extra-gcc-opts $(DESTDIR)$(libdir)
217 @case '${MFLAGS}' in *-[ik]*) x_on_err=0;; *-r*[ik]*) x_on_err=0;; *) x_on_err=1;; esac; \
218 for i in $(SUBDIRS); do \
219 echo "------------------------------------------------------------------------"; \
220 echo "== $(MAKE) $@ $(MFLAGS);"; \
221 echo " in $(shell pwd)/$$i"; \
222 echo "------------------------------------------------------------------------"; \
223 $(MAKE) --no-print-directory -C $$i $(INSTALL_STAGE) $(MFLAGS) $@; \
224 if [ $$? -eq 0 -o $$x_on_err -eq 0 ] ; then true; else exit 1; fi; \
227 # -----------------------------------------------------------------------------
228 # Making a binary distribution
230 # `dist' `binary-dist'
231 # Create a distribution tar file for this program. The tar file
232 # should be set up so that the file names in the tar file start with
233 # a subdirectory name which is the name of the package it is a
234 # distribution for. This name can include the version number.
236 # For example, the distribution tar file of GCC version 1.40 unpacks
237 # into a subdirectory named `gcc-1.40'.
239 # The easiest way to do this is to create a subdirectory
240 # appropriately named, use ln or cp to install the proper files in
241 # it, and then tar that subdirectory.
243 # The dist target should explicitly depend on all non-source files
244 # that are in the distribution, to make sure they are up to date in
245 # the distribution. See Making Releases.
247 # binary-dist is a GHC addition for binary distributions
251 -rm -rf $(BIN_DIST_DIR)
252 -$(RM) $(BIN_DIST_DIR).tar.gz
254 ifeq "$(TARGETPLATFORM)" "i386-unknown-mingw32"
257 $(MAKE) prefix=$(BIN_DIST_DIR) install
258 $(MAKE) prefix=$(BIN_DIST_DIR) install-docs
261 cd $(BIN_DIST_DIR) && ../distrib/prep-bin-dist-mingw
264 $(MKDIRHIER) $(BIN_DIST_DIR)/icons
265 cp distrib/hsicon.ico $(BIN_DIST_DIR)/icons
269 BinDistDirs = includes compiler docs rts
271 BIN_DIST_TOP= distrib/Makefile \
272 distrib/configure-bin.ac \
283 ifeq "$(darwin_TARGET_OS)" "1"
284 BIN_DIST_TOP+=mk/fix_install_names.sh
287 .PHONY: binary-dist-pre% binary-dist binary-pack
289 binary-dist:: binary-dist-pre
292 $(MKDIRHIER) $(BIN_DIST_DIR)/mk
293 echo 'include $$(TOP)/Makefile-vars' > $(BIN_DIST_DIR)/mk/boilerplate.mk
294 echo 'include $$(TOP)/mk/package.mk' > $(BIN_DIST_DIR)/mk/target.mk
295 echo 'include $$(TOP)/mk/install.mk' >> $(BIN_DIST_DIR)/mk/target.mk
296 echo 'include $$(TOP)/mk/recurse.mk' >> $(BIN_DIST_DIR)/mk/target.mk
297 echo '' > $(BIN_DIST_DIR)/mk/compat.mk
298 cp mk/package.mk $(BIN_DIST_DIR)/mk/
299 cp mk/install.mk $(BIN_DIST_DIR)/mk/
300 cp mk/recurse.mk $(BIN_DIST_DIR)/mk/
301 cp mk/fptools.css $(BIN_DIST_DIR)/mk/
302 $(MKDIRHIER) $(BIN_DIST_DIR)/lib/$(TARGETPLATFORM)
303 $(MKDIRHIER) $(BIN_DIST_DIR)/share
306 $(MAKE) -C gmp binary-dist DOING_BIN_DIST=YES
307 $(MAKE) -C includes binary-dist DOING_BIN_DIST=YES
308 $(MAKE) -C compiler binary-dist DOING_BIN_DIST=YES $(INSTALL_STAGE)
309 $(MAKE) -C rts binary-dist DOING_BIN_DIST=YES
310 $(MAKE) -C driver binary-dist DOING_BIN_DIST=YES
311 $(MAKE) -C utils binary-dist DOING_BIN_DIST=YES
312 $(MAKE) -C docs binary-dist DOING_BIN_DIST=YES
313 $(MAKE) -C libraries binary-dist DOING_BIN_DIST=YES
314 $(MAKE) -C libraries/Cabal/doc binary-dist DOING_BIN_DIST=YES
316 VARFILE=$(BIN_DIST_DIR)/Makefile-vars.in
319 @for i in $(BIN_DIST_TOP); do \
320 if test -f "$$i"; then \
321 echo cp $$i $(BIN_DIST_DIR); \
322 cp $$i $(BIN_DIST_DIR); \
325 @echo "Configuring the Makefile for this project..."
327 echo "package = ghc" >> $(VARFILE)
328 echo "version = $(ProjectVersion)" >> $(VARFILE)
329 echo "ProjectVersion = $(ProjectVersion)" >> $(VARFILE)
330 echo "HaveLibGmp = $(HaveLibGmp)" >> $(VARFILE)
331 echo "GhcLibsWithUnix = $(GhcLibsWithUnix)" >> $(VARFILE)
332 echo "GhcWithInterpreter = $(GhcWithInterpreter)" >> $(VARFILE)
333 echo "GhcHasReadline = $(GhcHasReadline)" >> $(VARFILE)
334 echo "BootingFromHc = $(BootingFromHc)" >> $(VARFILE)
335 echo "XMLDocWays = $(XMLDocWays)" >> $(VARFILE)
336 # We won't actually use xsltproc, but we need to know if it's "" or not
337 echo "XSLTPROC = $(XSLTPROC)" >> $(VARFILE)
338 echo "TARGETPLATFORM = $(TARGETPLATFORM)" >> $(VARFILE)
339 echo "HADDOCK_DOCS = $(HADDOCK_DOCS)" >> $(VARFILE)
341 cat distrib/Makefile-bin-vars.in >> $(VARFILE)
342 @echo "Generating a shippable configure script.."
343 $(MV) $(BIN_DIST_DIR)/configure-bin.ac $(BIN_DIST_DIR)/configure.ac
344 ( cd $(BIN_DIST_DIR); autoreconf )
347 # Tar up the distribution and build a manifest
348 binary-dist :: tar-binary-dist
350 .PHONY: tar-binary-dist
352 ( cd $(BIN_DIST_TOPDIR_ABS); tar cf - $(BIN_DIST_NAME) | bzip2 >$(BIN_DIST_TARBALL) )
353 ( cd $(BIN_DIST_TOPDIR_ABS); bunzip2 -c $(BIN_DIST_TARBALL) | tar tf - | sed "s/^ghc-$(ProjectVersion)/fptools/" | sort >bin-manifest-$(ProjectVersion) )
355 PUBLISH_FILES = $(BIN_DIST_TARBALL)
357 # Upload the distribution and documentation
359 WINDOWS_INSTALLER_BASE = ghc-$(ProjectVersion)-$(TARGETPLATFORM)
360 WINDOWS_INSTALLER = $(WINDOWS_INSTALLER_BASE)$(exeext)
362 PUBLISH_FILES += $(WINDOWS_INSTALLER)
364 binary-dist :: generate-windows-installer
366 .PHONY: generate-windows-installer
367 generate-windows-installer ::
368 $(SED) "s/@VERSION@/$(ProjectVersion)/" distrib/ghc.iss | $(ISCC) /O. /F$(WINDOWS_INSTALLER_BASE) -
371 # Upload the distribution and documentation
372 ifneq "$(PublishLocation)" ""
373 publish :: publish-binary-dist
376 .PHONY: publish-binary-dist
377 publish-binary-dist ::
378 @for f in $(PUBLISH_FILES); do \
379 for i in 0 1 2 3 4 5 6 7 8 9; do \
380 echo "Try $$i: $(PublishCp) $$f $(PublishLocation)/dist"; \
381 if $(PublishCp) $$f $(PublishLocation)/dist; then break; fi; \
385 # You need to first make binddisttest, and then run
386 # make publish 'prefix=$(BIN_DIST_INST_DIR)'
387 # for this to find the right place.
389 # We assume that Windows means Cygwin, as we can't just use docdir
390 # unchanged or rsync (really SSH?) thinks that c:/foo means /foo on
393 ifeq "$(Windows)" "YES"
394 PUBLISH_DOCDIR = $(shell cygpath --unix $(docdir))
396 PUBLISH_DOCDIR = $(docdir)
399 publish-binary-dist ::
400 $(PublishCp) -r $(PUBLISH_DOCDIR)/* $(PublishLocation)/docs
403 @echo "Mechanical and super-natty! Inspect the result and *if* happy; freeze, sell and get some sleep!"
405 # -----------------------------------------------------------------------------
406 # Building source distributions
413 # WARNING: `make dist' calls `make distclean' before tarring up the tree.
419 # Directory in which we're going to build the src dist
421 SRC_DIST_NAME=ghc-$(ProjectVersion)
422 SRC_DIST_DIR=$(shell pwd)/$(SRC_DIST_NAME)
425 # Files to include in source distributions
427 SRC_DIST_DIRS += mk docs distrib bindisttest $(filter-out docs distrib libraries/Cabal/doc,$(SUBDIRS))
429 configure.ac config.guess config.sub configure \
430 aclocal.m4 README ANNOUNCE HACKING LICENSE Makefile install-sh \
431 ghc.spec.in extra-gcc-opts.in VERSION boot
433 # -----------------------------------------------------------------------------
434 # Source distributions
436 # A source dist is built from a complete build tree, because we
437 # require some extra files not contained in a darcs checkout: the
438 # output from Happy and Alex, for example.
440 # The steps performed by 'make dist' are as follows:
441 # - create a complete link-tree of the current build tree in /tmp
442 # - run 'make distclean' on that tree
443 # - remove a bunch of other files that we know shouldn't be in the dist
444 # - tar up first the extralibs package, then the main source package
446 EXTRA_LIBS=$(patsubst %, $(SRC_DIST_NAME)/libraries/%, $(shell cat libraries/extra-packages))
448 SRC_DIST_TARBALL = ghc-$(ProjectVersion)-src.tar.bz2
449 SRC_DIST_EXTRALIBS_TARBALL = ghc-$(ProjectVersion)-src-extralibs.tar.bz2
452 echo $(ProjectVersion) >VERSION
457 $(RM) -rf $(SRC_DIST_DIR)
458 $(RM) $(SRC_DIST_NAME).tar.gz
459 mkdir $(SRC_DIST_DIR)
460 ( cd $(SRC_DIST_DIR) \
461 && for i in $(SRC_DIST_DIRS); do mkdir $$i; (cd $$i && lndir $(FPTOOLS_TOP_ABS)/$$i ); done \
462 && for i in $(SRC_DIST_FILES); do $(LN_S) $(FPTOOLS_TOP_ABS)/$$i .; done \
463 && $(MAKE) distclean \
464 && 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/ ; mv libraries/haskell-src/Language/Haskell/Parser.ly libraries/haskell-src/Language/Haskell/Parser.ly.source ; fi \
465 && $(RM) -rf compiler/stage[123] mk/build.mk \
466 && $(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 \
468 tar chf - $(EXTRA_LIBS) | bzip2 >$(FPTOOLS_TOP_ABS)/$(SRC_DIST_EXTRALIBS_TARBALL)
469 $(RM) -rf $(EXTRA_LIBS)
470 tar chf - $(SRC_DIST_NAME) 2>$src_log | bzip2 >$(FPTOOLS_TOP_ABS)/$(SRC_DIST_TARBALL)
472 # Upload the distribution(s)
473 # Retrying is to work around buggy firewalls that corrupt large file transfers
475 ifneq "$(PublishLocation)" ""
477 @for i in 0 1 2 3 4 5 6 7 8 9; do \
478 echo "Try $$i: $(PublishCp) $(SRC_DIST_EXTRALIBS_TARBALL) $(PublishLocation)/dist"; \
479 if $(PublishCp) $(SRC_DIST_EXTRALIBS_TARBALL) $(PublishLocation)/dist; then break; fi\
481 @for i in 0 1 2 3 4 5 6 7 8 9; do \
482 echo "Try $$i: $(PublishCp) $(SRC_DIST_TARBALL) $(PublishLocation)/dist"; \
483 if $(PublishCp) $(SRC_DIST_TARBALL) $(PublishLocation)/dist; then break; fi\
487 # -----------------------------------------------------------------------------
491 $(RM) -r ghc-$(ProjectVersion)
492 $(LN_S) . ghc-$(ProjectVersion)
493 $(FIND) ghc-$(ProjectVersion)/compiler \
494 ghc-$(ProjectVersion)/utils \
495 ghc-$(ProjectVersion)/compat \
496 ghc-$(ProjectVersion)/libraries -follow \
497 \( -name "*.hc" -o -name "*_hsc.[ch]" -o -name "*_stub.[ch]" \) -print > hc-files-to-go
498 for f in `$(FIND) ghc-$(ProjectVersion)/compiler ghc-$(ProjectVersion)/utils ghc-$(ProjectVersion)/libraries -name "*.hsc" -follow -print` ""; do \
499 if test "x$$f" != "x" && test -e `echo "$$f" | sed 's/hsc$$/hs/g'`; then \
500 echo `echo "$$f" | sed 's/hsc$$/hs/g' ` >> hc-files-to-go ; \
503 for f in `$(FIND) ghc-$(ProjectVersion)/compiler ghc-$(ProjectVersion)/rts -name "*.cmm" -follow -print` ""; do \
504 if test "x$$f" != "x"; then \
505 echo `echo "$$f" | sed 's/cmm$$/hc/g' ` >> hc-files-to-go ; \
508 echo ghc-$(ProjectVersion)/libraries/base/GHC/PrimopWrappers.hs >> hc-files-to-go
509 echo ghc-$(ProjectVersion)/compiler/parser/Parser.hs >> hc-files-to-go
510 echo ghc-$(ProjectVersion)/compiler/parser/ParserCore.hs >> hc-files-to-go
511 echo ghc-$(ProjectVersion)/compiler/main/ParsePkgConf.hs >> hc-files-to-go
512 echo ghc-$(ProjectVersion)/libraries/haskell-src/Language/Haskell/Parser.hs >> hc-files-to-go
513 tar czf ghc-$(ProjectVersion)-$(TARGETPLATFORM)-hc.tar.gz `cat hc-files-to-go`
515 # -----------------------------------------------------------------------------
518 CLEAN_FILES += hc-files-to-go *-hc.tar.gz
520 DIST_CLEAN_FILES += config.cache config.status mk/config.h mk/stamp-h \
521 ghc.spec docs/users_guide/ug-book.xml extra-gcc-opts
523 # don't clean config.mk: it's needed when cleaning stuff later on
524 LATE_DIST_CLEAN_FILES += mk/config.mk
526 # VERSION is shipped in a source dist
527 MAINTAINER_CLEAN_FILES += VERSION
530 $(RM) -rf autom4te.cache
533 $(MAKE) -C bindisttest $@
534 if test -d testsuite; then $(MAKE) -C testsuite $@; fi
536 # -----------------------------------------------------------------------------
538 # Turn off target.mk's rules for 'all', 'boot' and 'install'.
541 NO_INSTALL_TARGET=YES
543 include $(TOP)/mk/target.mk
545 # -----------------------------------------------------------------------------