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
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
255 $(MAKE) prefix=$(BIN_DIST_DIR) install-docs
258 cd $(BIN_DIST_DIR) && ../distrib/prep-bin-dist-mingw
261 $(MKDIRHIER) $(BIN_DIST_DIR)/icons
262 cp distrib/hsicon.ico $(BIN_DIST_DIR)/icons
266 BinDistDirs = includes compiler docs rts
268 BIN_DIST_TOP= distrib/Makefile \
269 distrib/configure-bin.ac \
280 ifeq "$(darwin_TARGET_OS)" "1"
281 BIN_DIST_TOP+=mk/fix_install_names.sh
284 .PHONY: binary-dist-pre% binary-dist binary-pack
286 binary-dist:: binary-dist-pre
289 $(MKDIRHIER) $(BIN_DIST_DIR)/mk
290 echo 'include $$(TOP)/Makefile-vars' > $(BIN_DIST_DIR)/mk/boilerplate.mk
291 echo 'include $$(TOP)/mk/package.mk' > $(BIN_DIST_DIR)/mk/target.mk
292 echo 'include $$(TOP)/mk/install.mk' >> $(BIN_DIST_DIR)/mk/target.mk
293 echo 'include $$(TOP)/mk/recurse.mk' >> $(BIN_DIST_DIR)/mk/target.mk
294 echo '' > $(BIN_DIST_DIR)/mk/compat.mk
295 cp mk/package.mk $(BIN_DIST_DIR)/mk/
296 cp mk/install.mk $(BIN_DIST_DIR)/mk/
297 cp mk/recurse.mk $(BIN_DIST_DIR)/mk/
298 cp mk/fptools.css $(BIN_DIST_DIR)/mk/
299 $(MKDIRHIER) $(BIN_DIST_DIR)/lib/$(TARGETPLATFORM)
300 $(MKDIRHIER) $(BIN_DIST_DIR)/share
303 $(MAKE) -C gmp binary-dist DOING_BIN_DIST=YES
304 $(MAKE) -C includes binary-dist DOING_BIN_DIST=YES
305 $(MAKE) -C compiler binary-dist DOING_BIN_DIST=YES $(INSTALL_STAGE)
306 $(MAKE) -C rts binary-dist DOING_BIN_DIST=YES
307 $(MAKE) -C driver binary-dist DOING_BIN_DIST=YES
308 $(MAKE) -C utils binary-dist DOING_BIN_DIST=YES
309 $(MAKE) -C docs binary-dist DOING_BIN_DIST=YES
310 $(MAKE) -C libraries binary-dist DOING_BIN_DIST=YES
311 $(MAKE) -C libraries/Cabal/doc binary-dist DOING_BIN_DIST=YES
313 VARFILE=$(BIN_DIST_DIR)/Makefile-vars.in
316 @for i in $(BIN_DIST_TOP); do \
317 if test -f "$$i"; then \
318 echo cp $$i $(BIN_DIST_DIR); \
319 cp $$i $(BIN_DIST_DIR); \
322 @echo "Configuring the Makefile for this project..."
324 echo "package = ghc" >> $(VARFILE)
325 echo "version = $(ProjectVersion)" >> $(VARFILE)
326 echo "ProjectVersion = $(ProjectVersion)" >> $(VARFILE)
327 echo "HaveLibGmp = $(HaveLibGmp)" >> $(VARFILE)
328 echo "GhcLibsWithUnix = $(GhcLibsWithUnix)" >> $(VARFILE)
329 echo "GhcWithInterpreter = $(GhcWithInterpreter)" >> $(VARFILE)
330 echo "GhcHasReadline = $(GhcHasReadline)" >> $(VARFILE)
331 echo "BootingFromHc = $(BootingFromHc)" >> $(VARFILE)
332 echo "XMLDocWays = $(XMLDocWays)" >> $(VARFILE)
333 # We won't actually use xsltproc, but we need to know if it's "" or not
334 echo "XSLTPROC = $(XSLTPROC)" >> $(VARFILE)
335 echo "TARGETPLATFORM = $(TARGETPLATFORM)" >> $(VARFILE)
336 echo "HADDOCK_DOCS = $(HADDOCK_DOCS)" >> $(VARFILE)
338 cat distrib/Makefile-bin-vars.in >> $(VARFILE)
339 @echo "Generating a shippable configure script.."
340 $(MV) $(BIN_DIST_DIR)/configure-bin.ac $(BIN_DIST_DIR)/configure.ac
341 ( cd $(BIN_DIST_DIR); autoreconf )
344 # Tar up the distribution and build a manifest
345 binary-dist :: tar-binary-dist
347 .PHONY: tar-binary-dist
349 ( cd $(BIN_DIST_TOPDIR_ABS); tar cf - $(BIN_DIST_NAME) | bzip2 >$(BIN_DIST_TARBALL) )
350 ( cd $(BIN_DIST_TOPDIR_ABS); bunzip2 -c $(BIN_DIST_TARBALL) | tar tf - | sed "s/^ghc-$(ProjectVersion)/fptools/" | sort >bin-manifest-$(ProjectVersion) )
352 PUBLISH_FILES = $(BIN_DIST_TARBALL)
354 # Upload the distribution and documentation
356 WINDOWS_INSTALLER_BASE = ghc-$(ProjectVersion)-$(TARGETPLATFORM)
357 WINDOWS_INSTALLER = $(WINDOWS_INSTALLER_BASE)$(exeext)
359 PUBLISH_FILES += $(WINDOWS_INSTALLER)
361 binary-dist :: generate-windows-installer
363 .PHONY: generate-windows-installer
364 generate-windows-installer ::
365 $(SED) "s/@VERSION@/$(ProjectVersion)/" distrib/ghc.iss | $(ISCC) /O. /F$(WINDOWS_INSTALLER_BASE) -
368 # Upload the distribution and documentation
369 ifneq "$(PublishLocation)" ""
370 publish :: publish-binary-dist
373 .PHONY: publish-binary-dist
374 publish-binary-dist ::
375 @for f in $(PUBLISH_FILES); do \
376 for i in 0 1 2 3 4 5 6 7 8 9; do \
377 echo "Try $$i: $(PublishCp) $$f $(PublishLocation)/dist"; \
378 if $(PublishCp) $$f $(PublishLocation)/dist; then break; fi; \
382 # You need to first make binddisttest, and then run
383 # make publish 'prefix=$(BIN_DIST_INST_DIR)'
384 # for this to find the right place.
386 # We assume that Windows means Cygwin, as we can't just use docdir
387 # unchanged or rsync (really SSH?) thinks that c:/foo means /foo on
390 ifeq "$(Windows)" "YES"
391 PUBLISH_DOCDIR = $(shell cygpath --unix $(docdir))
393 PUBLISH_DOCDIR = $(docdir)
396 publish-binary-dist ::
397 $(PublishCp) -r $(PUBLISH_DOCDIR)/* $(PublishLocation)/docs
400 @echo "Mechanical and super-natty! Inspect the result and *if* happy; freeze, sell and get some sleep!"
402 # -----------------------------------------------------------------------------
403 # Building source distributions
410 # WARNING: `make dist' calls `make distclean' before tarring up the tree.
416 # Directory in which we're going to build the src dist
418 SRC_DIST_NAME=ghc-$(ProjectVersion)
419 SRC_DIST_DIR=$(shell pwd)/$(SRC_DIST_NAME)
422 # Files to include in source distributions
424 SRC_DIST_DIRS += mk docs distrib bindisttest $(filter-out docs distrib libraries/Cabal/doc,$(SUBDIRS))
426 configure.ac config.guess config.sub configure \
427 aclocal.m4 README ANNOUNCE HACKING LICENSE Makefile install-sh \
428 ghc.spec.in extra-gcc-opts.in VERSION boot
430 # -----------------------------------------------------------------------------
431 # Source distributions
433 # A source dist is built from a complete build tree, because we
434 # require some extra files not contained in a darcs checkout: the
435 # output from Happy and Alex, for example.
437 # The steps performed by 'make dist' are as follows:
438 # - create a complete link-tree of the current build tree in /tmp
439 # - run 'make distclean' on that tree
440 # - remove a bunch of other files that we know shouldn't be in the dist
441 # - tar up first the extralibs package, then the main source package
443 EXTRA_LIBS=$(patsubst %, $(SRC_DIST_NAME)/libraries/%, $(shell cat libraries/extra-packages))
445 SRC_DIST_TARBALL = ghc-$(ProjectVersion)-src.tar.bz2
446 SRC_DIST_EXTRALIBS_TARBALL = ghc-$(ProjectVersion)-src-extralibs.tar.bz2
449 echo $(ProjectVersion) >VERSION
454 $(RM) -rf $(SRC_DIST_DIR)
455 $(RM) $(SRC_DIST_NAME).tar.gz
456 mkdir $(SRC_DIST_DIR)
457 ( cd $(SRC_DIST_DIR) \
458 && for i in $(SRC_DIST_DIRS); do mkdir $$i; (cd $$i && lndir $(FPTOOLS_TOP_ABS)/$$i ); done \
459 && for i in $(SRC_DIST_FILES); do $(LN_S) $(FPTOOLS_TOP_ABS)/$$i .; done \
460 && $(MAKE) distclean \
461 && 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 \
462 && $(RM) -rf compiler/stage[123] mk/build.mk \
463 && $(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 \
465 tar chf - $(EXTRA_LIBS) | bzip2 >$(FPTOOLS_TOP_ABS)/$(SRC_DIST_EXTRALIBS_TARBALL)
466 $(RM) -rf $(EXTRA_LIBS)
467 tar chf - $(SRC_DIST_NAME) 2>$src_log | bzip2 >$(FPTOOLS_TOP_ABS)/$(SRC_DIST_TARBALL)
469 # Upload the distribution(s)
470 # Retrying is to work around buggy firewalls that corrupt large file transfers
472 ifneq "$(PublishLocation)" ""
474 @for i in 0 1 2 3 4 5 6 7 8 9; do \
475 echo "Try $$i: $(PublishCp) $(SRC_DIST_EXTRALIBS_TARBALL) $(PublishLocation)/dist"; \
476 if $(PublishCp) $(SRC_DIST_EXTRALIBS_TARBALL) $(PublishLocation)/dist; then break; fi\
478 @for i in 0 1 2 3 4 5 6 7 8 9; do \
479 echo "Try $$i: $(PublishCp) $(SRC_DIST_TARBALL) $(PublishLocation)/dist"; \
480 if $(PublishCp) $(SRC_DIST_TARBALL) $(PublishLocation)/dist; then break; fi\
484 # -----------------------------------------------------------------------------
488 $(RM) -r ghc-$(ProjectVersion)
489 $(LN_S) . ghc-$(ProjectVersion)
490 $(FIND) ghc-$(ProjectVersion)/compiler \
491 ghc-$(ProjectVersion)/utils \
492 ghc-$(ProjectVersion)/compat \
493 ghc-$(ProjectVersion)/libraries -follow \
494 \( -name "*.hc" -o -name "*_hsc.[ch]" -o -name "*_stub.[ch]" \) -print > hc-files-to-go
495 for f in `$(FIND) ghc-$(ProjectVersion)/compiler ghc-$(ProjectVersion)/utils ghc-$(ProjectVersion)/libraries -name "*.hsc" -follow -print` ""; do \
496 if test "x$$f" != "x" && test -e `echo "$$f" | sed 's/hsc$$/hs/g'`; then \
497 echo `echo "$$f" | sed 's/hsc$$/hs/g' ` >> hc-files-to-go ; \
500 for f in `$(FIND) ghc-$(ProjectVersion)/compiler ghc-$(ProjectVersion)/rts -name "*.cmm" -follow -print` ""; do \
501 if test "x$$f" != "x"; then \
502 echo `echo "$$f" | sed 's/cmm$$/hc/g' ` >> hc-files-to-go ; \
505 echo ghc-$(ProjectVersion)/libraries/base/GHC/PrimopWrappers.hs >> hc-files-to-go
506 echo ghc-$(ProjectVersion)/compiler/parser/Parser.hs >> hc-files-to-go
507 echo ghc-$(ProjectVersion)/compiler/parser/ParserCore.hs >> hc-files-to-go
508 echo ghc-$(ProjectVersion)/compiler/main/ParsePkgConf.hs >> hc-files-to-go
509 echo ghc-$(ProjectVersion)/libraries/haskell-src/Language/Haskell/Parser.hs >> hc-files-to-go
510 tar czf ghc-$(ProjectVersion)-$(TARGETPLATFORM)-hc.tar.gz `cat hc-files-to-go`
512 # -----------------------------------------------------------------------------
515 CLEAN_FILES += hc-files-to-go *-hc.tar.gz
517 DIST_CLEAN_FILES += config.cache config.status mk/config.h mk/stamp-h \
518 ghc.spec docs/users_guide/ug-book.xml extra-gcc-opts
520 # don't clean config.mk: it's needed when cleaning stuff later on
521 LATE_DIST_CLEAN_FILES += mk/config.mk
523 # VERSION is shipped in a source dist
524 MAINTAINER_CLEAN_FILES += VERSION
527 $(RM) -rf autom4te.cache
530 $(MAKE) -C bindisttest $@
531 if test -d testsuite; then $(MAKE) -C testsuite $@; fi
533 # -----------------------------------------------------------------------------
535 # Turn off target.mk's rules for 'all', 'boot' and 'install'.
538 NO_INSTALL_TARGET=YES
540 include $(TOP)/mk/target.mk
542 # -----------------------------------------------------------------------------