相互参照警告が解決できない時に警告メッセージを表示するよう動作を変更
[makefiles.git] / latex_mk / latex.mk
index f157113..780e238 100644 (file)
@@ -6,23 +6,6 @@
 # 2. Makefileに変数TEXTARGETS と「include latex.mk」を記述する
 # 3. texソースファイルと同じディレクトリーで、make コマンドを実行する
 #
-# == 機能 ==
-# - 読み込むべき中間ファイルがないことや相互参照未定義の警告がある場合、LaTeX処理を最大4回繰り返す
-# - \includegraphics命令がTeXファイルに含まれる場合、グラフィックファイルを挿入
-#   -- 挿入されたグラフィックファイルが更新されたときは、処理を開始
-#   -- 挿入されたグラフィックファイルがないときは、処理を中止
-#   -- 挿入されたグラフィックファイルに対するバウンディング情報ファイル(.xbb)を作成
-# - \include、\input命令がTeXファイルに含まれる場合、TeXファイルを挿入
-#   -- 挿入されたTeXファイルが更新されたときは、処理を開始
-#   -- 挿入されたTeXファイルがないときは、処理を中止
-# - \makeindex命令が含まれる場合、mendexで索引を作成
-# - \bibliography命令が含まれる場合、BiBTeXで文献一覧を作成
-#
-# == 擬似ターゲット ==
-# - tex-clean: TeX中間ファイル(auxなど)を削除。ターゲットに.dviが含まれていないときは.dviファイルを削除
-# - tex-xbb-clean: バウンディング情報ファイル(.xbb)を削除
-# - tex-distclean: TeX中間ファイル、バウンディング情報ファイル、ターゲットファイル(PDF、.dvi)を削除
-#
 # === Makefile -- sample ===
 # TEXTARGETS := report.tex
 #
 #
 # include latex.mk
 
-# シェルスクリプトをデバッグするときは、DEBUGSH変数を設定してmakeを実行する
-# 例: DEBUGSH=1 make
-ifdef DEBUGSH
-  SHELL := /bin/sh -x
-endif
-
-.PHONY: tex-warn tex-xbb tex-clean tex-distclean
+######################################################################
+# 使用するシェルコマンドの定義
+######################################################################
 
 # シェルコマンド
 CAT := cat
@@ -56,14 +35,24 @@ BIBTEX := pbibtex
 MENDEX := mendex
 KPSEWHICH := kpsewhich
 
-#LaTeXオプション
+# LaTeXオプション
 LATEXFLAG ?=
 DVIPDFMXFLAG ?=
 EXTRACTBBFLAGS ?=
 BIBTEXFLAG ?=
 MENDEXFLAG ?=
 
-# .aux、.fls以外のTeX中間ファイルの拡張子
+# シェルコマンドをデバッグするときは、DEBUGSH変数を設定してmakeを実行する
+# 例: DEBUGSH=1 make
+ifdef DEBUGSH
+  SHELL := /bin/sh -x
+endif
+
+######################################################################
+# 拡張子
+######################################################################
+
+# .aux、.fls以外のLaTeX中間ファイルの拡張子
 #   .bbl: 文献リスト。作成方法はパターンルールで定義
 #   .glo: 用語集。\glossaryがあればTeX処理で生成
 #   .idx: 索引。\makeindexがあればTeX処理で生成
@@ -74,20 +63,29 @@ MENDEXFLAG ?=
 #   .toc: 目次。\tableofcontentsがあればTeX処理で生成
 LATEXINTEXT := .bbl .glo .idx .ind .lof .lot .out .toc
 
-# ログファイル
+# ログファイルの拡張子
 #   .log: TeXログ
 #   .ilg: 索引ログ
 #   .blg: BiBTeXログ
 LOGEXT := .log .ilg .blg
 
-ALLINTERFILES = $(addprefix *,.aux $(LATEXINTEXT) $(LOGEXT) .fls .d .*_prev)
+# すべてのTeX中間ファイルの拡張子
+ALLINTEXT := .aux .dvi $(LATEXINTEXT) $(LOGEXT) .fls .d .*_prev
+
+# 画像ファイルの拡張子
+GRAPHICSEXT := .pdf .eps .jpg .jpeg .png .bmp
 
 # make完了後、中間ファイルを残す
-.SECONDARY: $(wildcard ALLINTERFILES)
+.SECONDARY: $(foreach t,$(TEXTARGETS),$(addprefix $(basename $t),$(ALLINTEXT)))
 
 # ファイル名から拡張子を除いた部分
 BASE = $(basename $<)
 
+######################################################################
+# .dファイルの生成と読み込み
+# .dファイルには、LaTeX処理での依存関係が記述される
+######################################################################
+
 # .flsファイルから、INPUTファイルを取得。ただし、$TEXMFDISTのファイルを除く
 # 取得は、1回のmake実行につき1回だけ行われる
 INPUTFILES = $(INPUTFILESre)
@@ -108,39 +106,34 @@ OUTFILESre = $(eval OUTPUTFILES := \
     $(GREP) -v `$(KPSEWHICH) -expand-var '$$TEXMFROOT'` \
   ))))
 
-# $(BASE).texで読み込まれる中間ファイルを$(BASE).flsから取得する
+# $(BASE).texで読み込まれる中間ファイルを.flsから取得する
 # .idxは、.indへ置換
-LATEXINTERFILES = \
+LATEXINTFILES = \
   $(sort $(subst .idx,.ind, \
     $(filter $(addprefix $(BASE),$(LATEXINTEXT)),$(INPUTFILES) $(OUTPUTFILES)) \
   ))
 
-LATEXINTERFILES_PREV = $(addsuffix _prev,$(LATEXINTERFILES))
+LATEXINTFILES_PREV = $(addsuffix _prev,$(LATEXINTFILES))
 
-# TeXファイル - .tex
+# \includeや\inputで読み込まれるTeXファイルを.flsから取得する
 TEXFILES = $(filter %.tex,$(INPUTFILES))
 
-# 対応する画像ファイルの拡張子
-GRAPHICSEXT := .pdf .eps .jpg .jpeg .png .bmp
-
-# $(BASE).texで読み込まれる画像ファイルを取得する
+# \includegraphicsで読み込まれる画像ファイルを$(BASE).texと$(TEXFILES)、および.flsファイルから取得する
 GRAPHICFILES = $(GRAPHICFILESre)
 
 GRAPHICFILESre = $(eval GRAPHICFILES := \
   $(sort \
     $(shell \
       $(SED) -e '/^\s*%/d' -e 's/\([^\]\)\s*%.*/\1/g' $(BASE).tex $(TEXFILES) | \
-      $(SED) -e '/\\begin{verbatim}/,/\\end{verbatim}/d' | \
-      $(SED) -e 's/\\verb|[^|]*|//g' | \
+      $(SED) -e '/\\begin{verbatim}/,/\\end{verbatim}/d' -e 's/\\verb|[^|]*|//g' | \
       $(SED) -e 's/}/}%/g' | $(SED) -e 'y/}%/}\n/' | \
       $(SED) -n -e 's/.*\\includegraphics\(\[[^]]*\]\)\{0,1\}{\([^}]*\)}$$/\2/pg' \
     ) \
     $(filter $(addprefix %,$(GRAPHICSEXT)),$(INPUTFILES)) \
   ))
 
-# そのほかの読み込みファイル
-OTHERFILES = \
-  $(sort $(filter-out %.aux $(LATEXINTERFILES) $(TEXFILES) $(GRAPHICFILES),$(INPUTFILES)))
+# .flsから取得した、そのほかの読み込みファイル(.styなど)
+OTHERFILES = $(sort $(filter-out %.aux $(LATEXINTFILES) $(TEXFILES) $(GRAPHICFILES),$(INPUTFILES)))
 
 # \bibliography命令で読み込まれる文献データベースファイルをTeXファイルから検索する
 BIBDB = $(BIBDBre)
@@ -148,77 +141,38 @@ BIBDB = $(BIBDBre)
 BIBDBre = $(eval BIBDB := \
   $(addsuffix .bib,$(basename $(sort $(shell \
       $(SED) -e '/^\s*%/d' -e 's/\([^\]\)\s*%.*/\1/g' $(BASE).tex $(TEXFILES) | \
-      $(SED) -e '/\\begin{verbatim}/,/\\end{verbatim}/d' | \
+      $(SED) -e '/\\begin{verbatim}/,/\\end{verbatim}/d' -e 's/\\verb|[^|]*|//g' | \
       $(SED) -e 's/}/}%/g' | $(SED) -e 'y/}%/}\n/' | \
       $(SED) -n -e 's/.*\\bibliography\(\[[^]]*\]\)\{0,1\}{\([^}]*\)}$$/\2/pg' | \
       $(SED) -e 's/,/ /g' \
    )))))
 
-# LaTeX処理(コンパイル)
-LATEXCMD = $(LATEX) -interaction=batchmode $(LATEXFLAG) $(BASE).tex
-COMPILE.tex = $(ECHO) $(LATEXCMD); $(LATEXCMD) >/dev/null 2>&1 || ($(SED) -n -e '/^!/,/^$$/p' $(BASE).log; exit 1)
-
-# 相互参照未定義の警告
-WARN_UNDEFREF := 'There were undefined references\.'
-
-# LaTeX処理
-# ログファイルに警告がある場合は警告がなくなるまで、最大CNTで指定された回数分、処理を実行する
-CNT := 3
-COMPILES.tex = \
-  @(for i in `$(SEQ) 1 $(CNT)`; do \
-      if test -s $@ -a -s $(BASE).log; then \
-        $(GREP) -e $(WARN_UNDEFREF) $(BASE).log || exit 0; \
-      else \
-        $(ECHO) '$@ and/or $(BASE).log does not exist.'; \
-      fi; \
-      $(COMPILE.tex); \
-    done)
-
-# DVI -> PDF
-# 出力結果は.logファイルへ出力
-DVIPDFCMD = $(DVIPDFMX) $(DVIPDFMXFLAG) $(BASE).dvi
-COMPILE.dvi = \
-  $(ECHO) $(DVIPDFCMD); $(DVIPDFCMD) >>$(BASE).log 2>&1 || \
-  ($(SED) -n -e '/^Output written on toc_hyperref.dvi/,$$p' $(BASE).log; exit 1)
-
-# ターゲットファイルと必須ファイルを比較し、内容が異なる場合はターゲットファイルの内容を必須ファイルに置き換える
-CMPPREV = $(CMP) $@ $< || $(CP) -p -v $< $@
-
-######################################################################
-# .dファイルの生成と読み込み
-# .dファイルには、LaTeX処理での依存関係が記述される
-######################################################################
-
-# .flsファイル作成用の一時ディレクトリー
-FLSDIR := .fls.temp
-
-# $(BASE).flsファイルの作成
-FLSCMD = $(LATEX) -interaction=nonstopmode -recorder -output-directory=$(FLSDIR) $(BASE).tex
-
-GENERETE.fls = \
-  test ! -e $(FLSDIR) && $(MKDIR) $(FLSDIR); \
-  $(FLSCMD) 1>/dev/null 2>&1; \
-  $(SED) -e 's|$(FLSDIR)/||g' $(FLSDIR)/$(BASE).fls >$(BASE).fls; \
-  test -e $(BASE).fls && \
-    ($(ECHO) '$(BASE).fls is generated.'; $(RM) -r $(FLSDIR)) || \
-    ($(ECHO) '$(BASE).fls is not generated.' 1>&2; exit 1)
-
 # 依存関係を.dファイルに書き出す
 %.d: %.fls
+    # Makefile変数の展開
+       @$(foreach i,0 1,$(ECHO) "Makefiles variable -- LATEXINTFILES=$(LATEXINTFILES) TEXFILES=$(TEXFILES) GRAPHICFILES=$(GRAPHICFILES) BIBDB=$(BIBDB)" $(if $(filter 0,$i),>/dev/null);)
     # .dファイルの依存関係
        @$(ECHO) '$(BASE).d: $(BASE).tex $(BASE).fls' >$@
-    # \includeまたは\input命令で読み込まれるTeXファイルの依存関係
+    # 中間ファイルの依存関係
+       $(if $(sort $(LATEXINTFILES) $(BIBDB)),@( \
+      $(ECHO); \
+      $(ECHO) '# LaTeX Intermediate Files'; \
+      $(ECHO) '#'; \
+      $(ECHO) '# $$(COMPILE.tex) := $(LATEXCMD)'; \
+      $(ECHO) '# $$(COMPILES.tex) := $(subst $(COMPILE.tex),$(LATEXCMD),$(COMPILES.tex))'; \
+      $(ECHO) '#'; \
+      $(ECHO) '$(BASE).dvi:: $(sort $(LATEXINTFILES_PREV) $(if $(BIBDB),$(BASE).bbl_prev))'; \
+      $(ECHO) '        @$$(COMPILE.tex)'; \
+      $(ECHO); \
+      $(ECHO) '$(BASE).dvi:: $(BASE).aux'; \
+      $(ECHO) '        @$$(COMPILES.tex)'; \
+    ) >>$@)
+    # \includeや\inputで読み込まれるTeXファイルの依存関係
        $(if $(TEXFILES),@( \
       $(ECHO); \
       $(ECHO) '# Files called from \include or \input - .tex'; \
       $(ECHO) '$(BASE).aux: $(TEXFILES)'; \
     ) >>$@)
-    # そのほかのファイル(TEXMFROOT以外にあるスタイルファイルなど)の依存関係
-       $(if $(OTHERFILES),@( \
-      $(ECHO); \
-      $(ECHO) '# Other files'; \
-      $(ECHO) '$(BASE).aux: $(OTHERFILES)'; \
-    ) >>$@)
     # 画像ファイルの依存関係
        $(if $(GRAPHICFILES),@( \
       $(ECHO); \
@@ -230,21 +184,17 @@ GENERETE.fls = \
         $(ECHO) '$(BASE).aux: $(addsuffix .xbb,$(basename $(filter-out %.eps,$(GRAPHICFILES))))'; \
       ) \
     ) >>$@)
-    # 文献処理用ファイルの依存関係
+    # 文献リストファイルの依存関係
        $(if $(BIBDB),@( \
         $(ECHO); \
         $(ECHO) '# Bibliography files: .aux, BIBDB -> .bbl -> .div'; \
         $(ECHO) '$(BASE).bbl: $(BIBDB) $(BASE).tex'; \
       ) >>$@)
-    # 中間ファイルの依存関係
-       $(if $(sort $(LATEXINTERFILES) $(BIBDB)),@( \
-      $(ECHO); \
-      $(ECHO) '# LaTeX Intermediate Files'; \
-      $(ECHO) '$(BASE).dvi:: $(sort $(LATEXINTERFILES_PREV) $(if $(BIBDB),$(BASE).bbl_prev))'; \
-      $(ECHO) '        @$$(COMPILE.tex)'; \
+    # そのほかのファイル(TEXMFROOT以外にあるスタイルファイルなど)の依存関係
+       $(if $(OTHERFILES),@( \
       $(ECHO); \
-      $(ECHO) '$(BASE).dvi:: $(BASE).aux'; \
-      $(ECHO) '        @$$(COMPILES.tex)'; \
+      $(ECHO) '# Other files'; \
+      $(ECHO) '$(BASE).aux: $(OTHERFILES)'; \
     ) >>$@)
        @$(ECHO) '$@ is generated by scanning $(BASE).tex and $(BASE).fls.'
 
@@ -260,6 +210,52 @@ endif
 # TeX -> dvi -> PDF
 ######################################################################
 
+# LaTeX処理(コンパイル)
+LATEXCMD = $(LATEX) -interaction=batchmode $(LATEXFLAG) $(BASE).tex
+
+# エラー発生時、ログのエラー部分を行頭に「<TeXファイル名>:<行番号>:」を付けて表示する
+COMPILE.tex = \
+  $(ECHO) $(LATEXCMD); $(LATEXCMD) >/dev/null 2>&1 || \
+  ( \
+    $(SED) -n -e '/^!/,/^$$/p' $(BASE).log | \
+      $(SED) -e 's/.*\s*l\.\([0-9]*\)\s*.*/$(BASE).tex:\1: &/' 1>&2; \
+    exit 1)
+
+# 相互参照未定義の警告
+WARN_UNDEFREF := There were undefined references.
+
+# LaTeX処理
+# ログファイルに警告がある場合は警告がなくなるまで、最大CNTで指定された回数分、処理を実行する
+CNT := 3
+CNTMSG := $(LATEX) is run $(CNT) times, but there are still undefined references.
+
+COMPILES.tex = \
+  for i in `$(SEQ) 0 $(CNT)`; do \
+    if test $$i -lt $(CNT); then \
+      if $(GREP) -F "$(WARN_UNDEFREF)" $(BASE).log; then \
+        $(COMPILE.tex); \
+      else \
+        if test $$? -eq 1; then \
+          exit 0; \
+        else \
+          exit $$?; \
+        fi \
+      fi; \
+    else \
+      $(ECHO) "$(CNTMSG)"; \
+      $(SED) -n -e "/^LaTeX Warning:/,/^$$/p" $(BASE).log | \
+        $(SED) -e "s/.*\s*line \([0-9]*\)\s*.*/$(BASE).tex:\1: &/" 1>&2; \
+      exit 1; \
+    fi; \
+  done;
+
+# DVI -> PDF
+# 出力時のログは.logファイルへ追加出力
+DVIPDFCMD = $(DVIPDFMX) $(DVIPDFMXFLAG) $(BASE).dvi
+COMPILE.dvi = \
+  $(ECHO) $(DVIPDFCMD); $(DVIPDFCMD) >>$(BASE).log 2>&1 || \
+    ($(SED) -n -e '/^Output written on toc_hyperref.dvi/,$$p' $(BASE).log; exit 1)
+
 # TeX -> aux
 %.aux: %.tex
        @$(COMPILE.tex)
@@ -281,6 +277,26 @@ endif
 # ファイルリストファイル(.fls)作成
 ######################################################################
 
+# .flsファイル作成用の一時ディレクトリー
+FLSDIR := .fls.temp
+
+# $(BASE).flsファイルの作成
+FLSCMD = $(LATEX) -interaction=nonstopmode -recorder -output-directory=$(FLSDIR) $(BASE).tex
+
+GENERETE.fls = \
+  if test ! -e $(FLSDIR); then \
+    $(MKDIR) $(FLSDIR); \
+  fi; \
+  $(FLSCMD) 1>/dev/null 2>&1; \
+  $(SED) -e 's|$(FLSDIR)/||g' $(FLSDIR)/$(BASE).fls >$(BASE).fls; \
+  if test -e $(BASE).fls; then \
+    $(ECHO) '$(BASE).fls is generated.'; \
+    $(RM) -r $(FLSDIR); \
+  else \
+    $(ECHO) '$(BASE).fls is not generated.' 1>&2; \
+    exit 1; \
+  fi
+
 %.fls: %.tex
        @-$(GENERETE.fls)
 
@@ -288,6 +304,10 @@ endif
 # LaTeX中間ファイルを生成するパターンルール
 ######################################################################
 
+# ターゲットファイルと必須ファイルを比較し、
+# 内容が異なる場合はターゲットファイルの内容を必須ファイルに置き換える
+CMPPREV = $(CMP) $< $@ && $(ECHO) '$@ is up to date.' || $(CP) -p -v $< $@
+
 # 図リスト
 %.lof: %.tex
        @$(MAKE) -s $(BASE).aux
@@ -323,7 +343,7 @@ endif
 # 索引用中間ファイル作成コマンド
 MENDEXCMD = $(MENDEX) $(MENDEXFLAG) $(BASE).idx
 
-COMPILE.idx = $(ECHO) $(MENDEXCMD); $(MENDEXCMD) >/dev/null 2>&1 || ($(CAT) $(BASE).ilg; exit 1)
+COMPILE.idx = $(ECHO) $(MENDEXCMD); $(MENDEXCMD) >/dev/null 2>&1 || ($(CAT) $(BASE).ilg 1>&2; exit 1)
 
 # .tex -> .idx
 %.idx: %.tex
@@ -345,7 +365,7 @@ COMPILE.idx = $(ECHO) $(MENDEXCMD); $(MENDEXCMD) >/dev/null 2>&1 || ($(CAT) $(BA
 # 文献リスト用中間ファイル作成コマンド
 BIBTEXCMD = $(BIBTEX) $(BIBTEXFLAG) $(BASE).aux
 
-COMPILE.bib = $(ECHO) $(BIBTEXCMD); $(BIBTEXCMD) >/dev/null 2>&1 || ($(CAT) $(BASE).blg; exit 1)
+COMPILE.bib = $(ECHO) $(BIBTEXCMD); $(BIBTEXCMD) >/dev/null 2>&1 || ($(CAT) $(BASE).blg 1>&2; exit 1)
 
 # TeX -> .aux -> .bib
 %.bbl: %.tex
@@ -379,7 +399,7 @@ COMPILE.bib = $(ECHO) $(BIBTEXCMD); $(BIBTEXCMD) >/dev/null 2>&1 || ($(CAT) $(BA
 
 # 警告
 tex-warn:
-       @$(ECHO) "check current directory, or set TEXTARGET in Makefile."
+       @($(ECHO) "check current directory, or set TEXTARGET in Makefile." 1>&2)
 
 # すべての画像ファイルに対してextractbbを実行
 tex-xbb:
@@ -387,7 +407,7 @@ tex-xbb:
 
 # 中間ファイルの削除
 tex-clean:
-       $(RM) $(ALLINTERFILES)
+       $(RM) $(addprefix *,$(ALLINTEXT))
        $(RM) -r $(FLSDIR)
 ifeq (,$(filter %.dvi,$(TEXTARGETS)))
        $(RM) *.dvi