1# -*- makefile-gmake -*-
2
3.PHONY: all
4all: notmuch notmuch-shared build-man build-info ruby-bindings python-cffi-bindings
5ifeq ($(MAKECMDGOALS),)
6ifeq ($(shell cat .first-build-message 2>/dev/null),)
7	@NOTMUCH_FIRST_BUILD=1 $(MAKE) --no-print-directory all
8	@echo ""
9	@echo "Compilation of notmuch is now complete. You can install notmuch with:"
10	@echo ""
11	@echo "	make install"
12	@echo ""
13	@echo "Note that depending on the prefix to which you are installing"
14	@echo "you may need root permission (such as \"sudo make install\")."
15	@echo "See \"./configure --help\" for help on setting an alternate prefix."
16	@echo Printed > .first-build-message
17endif
18endif
19
20# Depend (also) on the file 'version'. In case of ifeq ($(IS_GIT),yes)
21# this file may already have been updated.
22version.stamp: $(srcdir)/version.txt
23	echo $(VERSION) > $@
24
25$(TAR_FILE):
26	if git tag -v $(UPSTREAM_TAG) >/dev/null 2>&1; then \
27           ref=$(UPSTREAM_TAG); \
28        else \
29           ref="HEAD" ; \
30	   echo "Warning: No signed tag for $(VERSION)"; \
31	fi ; \
32	git archive --format=tar --prefix=$(PACKAGE)-$(VERSION)/ $$ref > $(TAR_FILE).tmp
33	echo $(VERSION) > version.txt.tmp
34	ct=`git --no-pager log -1 --pretty=format:%ct $$ref` ; \
35	tar --owner root --group root --append -f $(TAR_FILE).tmp \
36		--transform s_^_$(PACKAGE)-$(VERSION)/_  \
37		--transform 's_.tmp$$__' --mtime=@$$ct version.txt.tmp
38	rm version.txt.tmp
39	xz -C sha256 -9 < $(TAR_FILE).tmp > $(TAR_FILE)
40	@echo "Source is ready for release in $(TAR_FILE)"
41
42$(SHA256_FILE): $(TAR_FILE)
43	sha256sum $^ | gpg --clear-sign --output $@ -
44
45$(DETACHED_SIG_FILE): $(TAR_FILE)
46	gpg --armor --detach-sign $^
47
48.PHONY: dist
49dist: $(TAR_FILE)
50
51.PHONY: update-versions
52
53update-versions:
54	sed -i -e "s/^__VERSION__[[:blank:]]*=.*$$/__VERSION__ = \'${VERSION}\'/" \
55	    -e "s/^SOVERSION[[:blank:]]*=.*$$/SOVERSION = \'${LIBNOTMUCH_VERSION_MAJOR}\'/" \
56	    ${PV_FILE}
57	cp version.txt bindings/python-cffi
58
59# We invoke make recursively only to force ordering of our phony
60# targets in the case of parallel invocation of make (-j).
61#
62# We carefully ensure that our VERSION variable is passed down to any
63# sub-ordinate make invocations (which won't otherwise know that they
64# are part of the release and need to take the version from the
65# version file).
66.PHONY: release
67release: verify-source-tree-and-version
68	$(MAKE) VERSION=$(VERSION) verify-newer
69	$(MAKE) VERSION=$(VERSION) clean
70	$(MAKE) VERSION=$(VERSION) sphinx-html
71	$(MAKE) VERSION=$(VERSION) test
72	git tag -s -m "$(PACKAGE) $(VERSION) release" $(UPSTREAM_TAG)
73	$(MAKE) VERSION=$(VERSION) $(SHA256_FILE) $(DETACHED_SIG_FILE)
74	ln -sf $(TAR_FILE) $(DEB_TAR_FILE)
75	pristine-tar commit $(DEB_TAR_FILE) $(UPSTREAM_TAG)
76	mkdir -p releases
77	mv $(TAR_FILE) $(SHA256_FILE) $(DETACHED_SIG_FILE) releases
78	$(MAKE) VERSION=$(VERSION) release-message > $(PACKAGE)-$(VERSION).announce
79ifeq ($(REALLY_UPLOAD),yes)
80	git push origin $(VERSION) release pristine-tar
81	cd releases && scp $(TAR_FILE) $(SHA256_FILE) $(DETACHED_SIG_FILE) $(RELEASE_HOST):$(RELEASE_DIR)
82	ssh $(RELEASE_HOST) "rm -f $(RELEASE_DIR)/LATEST-$(PACKAGE)-* ; ln -s $(TAR_FILE) $(RELEASE_DIR)/LATEST-$(TAR_FILE)"
83	rsync --verbose --delete --recursive doc/_build/html/ $(RELEASE_HOST):$(DOC_DIR)
84endif
85	@echo "Please send a release announcement using $(PACKAGE)-$(VERSION).announce as a template."
86
87.PHONY: pre-release
88pre-release:
89	$(MAKE) VERSION=$(VERSION) clean
90	$(MAKE) VERSION=$(VERSION) test
91	git tag -s -m "$(PACKAGE) $(VERSION) release" $(UPSTREAM_TAG)
92	$(MAKE) VERSION=$(VERSION) $(SHA256_FILE) $(DETACHED_SIG_FILE)
93	ln -sf $(TAR_FILE) $(DEB_TAR_FILE)
94	pristine-tar commit $(DEB_TAR_FILE) $(UPSTREAM_TAG)
95	mkdir -p releases
96	mv $(TAR_FILE) $(DEB_TAR_FILE) releases
97
98.PHONY: debian-snapshot
99debian-snapshot:
100	make VERSION=$(VERSION) clean
101	RETVAL=0 &&						\
102	  TMPFILE=$$(mktemp /tmp/notmuch.XXXXXX) &&		\
103	  cp debian/changelog $${TMPFILE} &&			\
104	  (EDITOR=/bin/true dch -b -v $(VERSION)+1		\
105	    -D UNRELEASED 'test build, not for upload' &&	\
106	  echo '3.0 (native)' > debian/source/format &&		\
107	  debuild -us -uc); RETVAL=$$?				\
108	  mv -f $${TMPFILE} debian/changelog;			\
109	  echo '3.0 (quilt)' > debian/source/format;		\
110	  exit $$RETVAL
111
112.PHONY: release-message
113release-message:
114	@echo "To: notmuch@notmuchmail.org"
115	@echo "Subject: $(PACKAGE) release $(VERSION) now available"
116	@echo ""
117	@echo "Where to obtain notmuch $(VERSION)"
118	@echo "==========================="
119	@echo "  $(RELEASE_URL)/$(TAR_FILE)"
120	@echo ""
121	@echo "Which can be verified with:"
122	@echo ""
123	@echo "  $(RELEASE_URL)/$(SHA256_FILE)"
124	@echo -n "  "
125	@cat releases/$(SHA256_FILE)
126	@echo ""
127	@echo "  $(RELEASE_URL)/$(DETACHED_SIG_FILE)"
128	@echo "  (signed by `getent passwd "$$USER" | cut -d: -f 5 | cut -d, -f 1`)"
129	@echo ""
130	@echo "What's new in notmuch $(VERSION)"
131	@echo "========================="
132	@sed -ne '/^[Nn]otmuch $(VERSION)/{n;n;b NEWS}; d; :NEWS /^===/q; {p;n;b NEWS}' < NEWS | head -n -2
133	@echo ""
134	@echo "What is notmuch"
135	@echo "==============="
136	@echo "Notmuch is a system for indexing, searching, reading, and tagging"
137	@echo "large collections of email messages in maildir or mh format. It uses"
138	@echo "the Xapian library to provide fast, full-text search with a convenient"
139	@echo "search syntax."
140	@echo ""
141	@echo "For more about notmuch, see https://notmuchmail.org"
142
143# This is a chain of dependencies rather than a simple list simply to
144# avoid the messages getting interleaved in the case of a parallel
145# make invocation.
146.PHONY: verify-source-tree-and-version
147verify-source-tree-and-version: verify-no-dirty-code
148
149.PHONY: verify-no-dirty-code
150verify-no-dirty-code: release-checks
151ifeq ($(IS_GIT),yes)
152	@printf "Checking that source tree is clean..."
153ifneq ($(shell git --git-dir=${srcdir}/.git ls-files -m),)
154	@echo "No"
155	@echo "The following files have been modified since the most recent git commit:"
156	@echo ""
157	@git --git-dir=${srcdir}/.git ls-files -m
158	@echo ""
159	@echo "The release will be made from the committed state, but perhaps you meant"
160	@echo "to commit this code first? Please clean this up to make it more clear."
161	@false
162else
163	@echo "Good"
164endif
165endif
166
167.PHONY: release-checks
168release-checks:
169	devel/release-checks.sh
170
171.PHONY: verify-newer
172verify-newer:
173	@echo -n "Checking that no $(VERSION) release already exists..."
174	@wget -q --no-check-certificate -O /dev/null $(RELEASE_URL)/$(TAR_FILE) ; \
175	case $$? in \
176          8) echo "Good." ;; \
177          0) echo "Ouch."; \
178	     echo "Found: $(RELEASE_URL)/$(TAR_FILE)"; \
179	     echo "Refusing to replace an existing release."; \
180	     echo "Don't forget to update \"version\" as described in RELEASING before release." ; \
181	     false ;; \
182	  *) echo "An unexpected error occurred"; \
183	     false;; esac
184
185# The user has not set any verbosity, default to quiet mode and inform the
186# user how to enable verbose compiles.
187ifeq ($(V),)
188quiet_DOC := "Use \"$(MAKE) V=1\" to see the verbose compile lines.\n"
189quiet = @printf $(quiet_DOC)$(eval quiet_DOC:=)"$(1) $(or $(2),$@)\n"; $($(word 1, $(1)))
190endif
191# The user has explicitly enabled quiet compilation.
192ifeq ($(V),0)
193quiet = @printf "$(1) $(or $(2),$@)\n"; $($(word 1, $(1)))
194endif
195# Otherwise, print the full command line.
196quiet ?= $($(word 1, $(1)))
197
198%.o: %.cc $(global_deps)
199	@mkdir -p $(patsubst %/.,%,.deps/$(@D))
200	$(call quiet,CXX $(CPPFLAGS) $(CXXFLAGS)) -c $(FINAL_CXXFLAGS) $< -o $@ -MD -MP -MF .deps/$*.d
201
202%.o: %.c $(global_deps)
203	@mkdir -p $(patsubst %/.,%,.deps/$(@D))
204	$(call quiet,CC $(CPPFLAGS) $(CFLAGS)) -c $(FINAL_CFLAGS) $< -o $@ -MD -MP -MF .deps/$*.d
205
206CPPCHECK=cppcheck
207.stamps/cppcheck/%: %
208	@mkdir -p $(@D)
209	$(call quiet,CPPCHECK,$<) --template=gcc --error-exitcode=1 --quiet $<
210	@touch $@
211
212CLEAN := $(CLEAN) .stamps
213
214.PHONY : clean
215clean:
216	rm -rf $(CLEAN)
217
218.PHONY: distclean
219distclean: clean
220	rm -rf $(DISTCLEAN)
221
222.PHONY: dataclean
223dataclean: distclean
224	rm -rf $(DATACLEAN)
225
226notmuch_client_srcs =		\
227	$(notmuch_compat_srcs)	\
228	command-line-arguments.c\
229	debugger.c		\
230	status.c		\
231	gmime-filter-reply.c	\
232	hooks.c			\
233	notmuch.c		\
234	notmuch-client-init.c	\
235	notmuch-compact.c	\
236	notmuch-config.c	\
237	notmuch-count.c		\
238	notmuch-dump.c		\
239	notmuch-insert.c	\
240	notmuch-new.c		\
241	notmuch-reindex.c       \
242	notmuch-reply.c		\
243	notmuch-restore.c	\
244	notmuch-search.c	\
245	notmuch-setup.c		\
246	notmuch-show.c		\
247	notmuch-tag.c		\
248	notmuch-time.c		\
249	sprinter-json.c		\
250	sprinter-sexp.c		\
251	sprinter-text.c		\
252	query-string.c		\
253	mime-node.c		\
254	tag-util.c
255
256notmuch_client_modules = $(notmuch_client_srcs:.c=.o)
257
258notmuch.o: version.stamp
259
260notmuch: $(notmuch_client_modules) lib/libnotmuch.a util/libnotmuch_util.a parse-time-string/libparse-time-string.a
261	$(call quiet,CXX $(CFLAGS)) $^ $(FINAL_LIBNOTMUCH_LDFLAGS) -o $@
262
263notmuch-shared: $(notmuch_client_modules) lib/$(LINKER_NAME)
264	$(call quiet,$(FINAL_NOTMUCH_LINKER) $(CFLAGS)) $(notmuch_client_modules) $(FINAL_NOTMUCH_LDFLAGS) -o $@
265
266.PHONY: install
267install: all install-man install-info
268	mkdir -p "$(DESTDIR)$(prefix)/bin/"
269	install notmuch-shared "$(DESTDIR)$(prefix)/bin/notmuch"
270ifeq ($(MAKECMDGOALS), install)
271	@echo ""
272	@echo "Notmuch is now installed to $(DESTDIR)$(prefix)"
273	@echo ""
274	@echo "New users should simply run \"notmuch\" to be guided"
275	@echo "through the process of configuring notmuch and creating"
276	@echo "a database of existing email messages. The \"notmuch\""
277	@echo "command will also offer some sample search commands."
278ifeq ($(WITH_EMACS), 1)
279	@echo ""
280	@echo "Beyond the command-line interface, notmuch also offers"
281	@echo "a full-featured interface for reading and writing mail"
282	@echo "within emacs. To use this, each user should add the"
283	@echo "following line to the ~/.emacs file:"
284	@echo ""
285	@echo "	(require 'notmuch)"
286	@echo ""
287	@echo "And then run emacs as \"emacs -f notmuch\" or invoke"
288	@echo "the command \"M-x notmuch\" from within emacs."
289endif
290endif
291
292SRCS  := $(SRCS) $(notmuch_client_srcs)
293CLEAN := $(CLEAN) notmuch notmuch-shared $(notmuch_client_modules)
294CLEAN := $(CLEAN) version.stamp notmuch-*.tar.gz.tmp
295CLEAN := $(CLEAN) .deps
296
297DISTCLEAN := $(DISTCLEAN) .first-build-message Makefile.config sh.config sphinx.config
298
299CPPCHECK_STAMPS := $(SRCS:%=.stamps/cppcheck/%)
300.PHONY: cppcheck
301ifeq ($(HAVE_CPPCHECK),1)
302cppcheck: ${CPPCHECK_STAMPS}
303else
304cppcheck:
305	@echo "No cppcheck found during configure; skipping static checking"
306endif
307
308
309DEPS := $(SRCS:%.c=.deps/%.d)
310DEPS := $(DEPS:%.cc=.deps/%.d)
311-include $(DEPS)
312
313.SUFFIXES: # Delete the default suffixes. Old-Fashioned Suffix Rules not used.
314