| Michael Witten | ced465c | 2011-04-02 21:46:09 +0000 | [diff] [blame] | 1 | # This allows us to work with the newline character: | 
|  | 2 | define newline | 
|  | 3 |  | 
|  | 4 |  | 
|  | 5 | endef | 
|  | 6 | newline := $(newline) | 
|  | 7 |  | 
|  | 8 | # nl-escape | 
|  | 9 | # | 
|  | 10 | # Usage: escape = $(call nl-escape[,escape]) | 
|  | 11 | # | 
|  | 12 | # This is used as the common way to specify | 
|  | 13 | # what should replace a newline when escaping | 
|  | 14 | # newlines; the default is a bizarre string. | 
|  | 15 | # | 
|  | 16 | nl-escape = $(or $(1),m822df3020w6a44id34bt574ctac44eb9f4n) | 
|  | 17 |  | 
|  | 18 | # escape-nl | 
|  | 19 | # | 
|  | 20 | # Usage: escaped-text = $(call escape-nl,text[,escape]) | 
|  | 21 | # | 
|  | 22 | # GNU make's $(shell ...) function converts to a | 
|  | 23 | # single space each newline character in the output | 
|  | 24 | # produced during the expansion; this may not be | 
|  | 25 | # desirable. | 
|  | 26 | # | 
|  | 27 | # The only solution is to change each newline into | 
|  | 28 | # something that won't be converted, so that the | 
|  | 29 | # information can be recovered later with | 
|  | 30 | # $(call unescape-nl...) | 
|  | 31 | # | 
|  | 32 | escape-nl = $(subst $(newline),$(call nl-escape,$(2)),$(1)) | 
|  | 33 |  | 
|  | 34 | # unescape-nl | 
|  | 35 | # | 
|  | 36 | # Usage: text = $(call unescape-nl,escaped-text[,escape]) | 
|  | 37 | # | 
|  | 38 | # See escape-nl. | 
|  | 39 | # | 
|  | 40 | unescape-nl = $(subst $(call nl-escape,$(2)),$(newline),$(1)) | 
|  | 41 |  | 
|  | 42 | # shell-escape-nl | 
|  | 43 | # | 
|  | 44 | # Usage: $(shell some-command | $(call shell-escape-nl[,escape])) | 
|  | 45 | # | 
|  | 46 | # Use this to escape newlines from within a shell call; | 
|  | 47 | # the default escape is a bizarre string. | 
|  | 48 | # | 
|  | 49 | # NOTE: The escape is used directly as a string constant | 
|  | 50 | #       in an `awk' program that is delimited by shell | 
|  | 51 | #       single-quotes, so be wary of the characters | 
|  | 52 | #       that are chosen. | 
|  | 53 | # | 
|  | 54 | define shell-escape-nl | 
|  | 55 | awk 'NR==1 {t=$$0} NR>1 {t=t "$(nl-escape)" $$0} END {printf t}' | 
|  | 56 | endef | 
|  | 57 |  | 
|  | 58 | # shell-unescape-nl | 
|  | 59 | # | 
|  | 60 | # Usage: $(shell some-command | $(call shell-unescape-nl[,escape])) | 
|  | 61 | # | 
|  | 62 | # Use this to unescape newlines from within a shell call; | 
|  | 63 | # the default escape is a bizarre string. | 
|  | 64 | # | 
|  | 65 | # NOTE: The escape is used directly as an extended regular | 
|  | 66 | #       expression constant in an `awk' program that is | 
|  | 67 | #       delimited by shell single-quotes, so be wary | 
|  | 68 | #       of the characters that are chosen. | 
|  | 69 | # | 
|  | 70 | # (The bash shell has a bug where `{gsub(...),...}' is | 
|  | 71 | #  misinterpreted as a brace expansion; this can be | 
|  | 72 | #  overcome by putting a space between `{' and `gsub'). | 
|  | 73 | # | 
|  | 74 | define shell-unescape-nl | 
|  | 75 | awk 'NR==1 {t=$$0} NR>1 {t=t "\n" $$0} END { gsub(/$(nl-escape)/,"\n",t); printf t }' | 
|  | 76 | endef | 
|  | 77 |  | 
|  | 78 | # escape-for-shell-sq | 
|  | 79 | # | 
|  | 80 | # Usage: embeddable-text = $(call escape-for-shell-sq,text) | 
|  | 81 | # | 
|  | 82 | # This function produces text that is suitable for | 
|  | 83 | # embedding in a shell string that is delimited by | 
|  | 84 | # single-quotes. | 
|  | 85 | # | 
|  | 86 | escape-for-shell-sq =  $(subst ','\'',$(1)) | 
|  | 87 |  | 
|  | 88 | # shell-sq | 
|  | 89 | # | 
|  | 90 | # Usage: single-quoted-and-escaped-text = $(call shell-sq,text) | 
|  | 91 | # | 
|  | 92 | shell-sq = '$(escape-for-shell-sq)' | 
|  | 93 |  | 
|  | 94 | # shell-wordify | 
|  | 95 | # | 
|  | 96 | # Usage: wordified-text = $(call shell-wordify,text) | 
|  | 97 | # | 
|  | 98 | # For instance: | 
|  | 99 | # | 
|  | 100 | #  |define text | 
|  | 101 | #  |hello | 
|  | 102 | #  |world | 
|  | 103 | #  |endef | 
|  | 104 | #  | | 
|  | 105 | #  |target: | 
|  | 106 | #  |	echo $(call shell-wordify,$(text)) | 
|  | 107 | # | 
|  | 108 | # At least GNU make gets confused by expanding a newline | 
|  | 109 | # within the context of a command line of a makefile rule | 
|  | 110 | # (this is in constrast to a `$(shell ...)' function call, | 
|  | 111 | # which can handle it just fine). | 
|  | 112 | # | 
|  | 113 | # This function avoids the problem by producing a string | 
|  | 114 | # that works as a shell word, regardless of whether or | 
|  | 115 | # not it contains a newline. | 
|  | 116 | # | 
|  | 117 | # If the text to be wordified contains a newline, then | 
|  | 118 | # an intrictate shell command substitution is constructed | 
|  | 119 | # to render the text as a single line; when the shell | 
|  | 120 | # processes the resulting escaped text, it transforms | 
|  | 121 | # it into the original unescaped text. | 
|  | 122 | # | 
|  | 123 | # If the text does not contain a newline, then this function | 
|  | 124 | # produces the same results as the `$(shell-sq)' function. | 
|  | 125 | # | 
|  | 126 | shell-wordify = $(if $(findstring $(newline),$(1)),$(_sw-esc-nl),$(shell-sq)) | 
|  | 127 | define _sw-esc-nl | 
|  | 128 | "$$(echo $(call escape-nl,$(shell-sq),$(2)) | $(call shell-unescape-nl,$(2)))" | 
|  | 129 | endef | 
|  | 130 |  | 
|  | 131 | # is-absolute | 
|  | 132 | # | 
|  | 133 | # Usage: bool-value = $(call is-absolute,path) | 
|  | 134 | # | 
|  | 135 | is-absolute = $(shell echo $(shell-sq) | grep ^/ -q && echo y) | 
|  | 136 |  | 
|  | 137 | # lookup | 
|  | 138 | # | 
|  | 139 | # Usage: absolute-executable-path-or-empty = $(call lookup,path) | 
|  | 140 | # | 
|  | 141 | # (It's necessary to use `sh -c' because GNU make messes up by | 
|  | 142 | #  trying too hard and getting things wrong). | 
|  | 143 | # | 
|  | 144 | lookup = $(call unescape-nl,$(shell sh -c $(_l-sh))) | 
|  | 145 | _l-sh = $(call shell-sq,command -v $(shell-sq) | $(call shell-escape-nl,)) | 
|  | 146 |  | 
|  | 147 | # is-executable | 
|  | 148 | # | 
|  | 149 | # Usage: bool-value = $(call is-executable,path) | 
|  | 150 | # | 
|  | 151 | # (It's necessary to use `sh -c' because GNU make messes up by | 
|  | 152 | #  trying too hard and getting things wrong). | 
|  | 153 | # | 
|  | 154 | is-executable = $(call _is-executable-helper,$(shell-sq)) | 
|  | 155 | _is-executable-helper = $(shell sh -c $(_is-executable-sh)) | 
|  | 156 | _is-executable-sh = $(call shell-sq,test -f $(1) -a -x $(1) && echo y) | 
|  | 157 |  | 
|  | 158 | # get-executable | 
|  | 159 | # | 
|  | 160 | # Usage: absolute-executable-path-or-empty = $(call get-executable,path) | 
|  | 161 | # | 
|  | 162 | # The goal is to get an absolute path for an executable; | 
|  | 163 | # the `command -v' is defined by POSIX, but it's not | 
|  | 164 | # necessarily very portable, so it's only used if | 
|  | 165 | # relative path resolution is requested, as determined | 
|  | 166 | # by the presence of a leading `/'. | 
|  | 167 | # | 
|  | 168 | get-executable = $(if $(1),$(if $(is-absolute),$(_ge-abspath),$(lookup))) | 
|  | 169 | _ge-abspath = $(if $(is-executable),$(1)) | 
|  | 170 |  | 
|  | 171 | # get-supplied-or-default-executable | 
|  | 172 | # | 
|  | 173 | # Usage: absolute-executable-path-or-empty = $(call get-executable-or-default,variable,default) | 
|  | 174 | # | 
|  | 175 | define get-executable-or-default | 
|  | 176 | $(if $($(1)),$(call _ge_attempt,$($(1)),$(1)),$(call _ge_attempt,$(2))) | 
|  | 177 | endef | 
|  | 178 | _ge_attempt = $(or $(get-executable),$(_gea_warn),$(call _gea_err,$(2))) | 
|  | 179 | _gea_warn = $(warning The path '$(1)' is not executable.) | 
|  | 180 | _gea_err  = $(if $(1),$(error Please set '$(1)' appropriately)) | 
| Michael Witten | 7fbd065 | 2011-04-12 20:27:59 +0000 | [diff] [blame] | 181 |  | 
|  | 182 | # try-cc | 
| Jiri Olsa | 28d213b | 2012-10-09 17:50:01 +0200 | [diff] [blame] | 183 | # Usage: option = $(call try-cc, source-to-build, cc-options, msg) | 
|  | 184 | ifndef V | 
|  | 185 | TRY_CC_OUTPUT= > /dev/null 2>&1 | 
| Jiri Olsa | 28d213b | 2012-10-09 17:50:01 +0200 | [diff] [blame] | 186 | endif | 
| Namhyung Kim | cf3aa10 | 2012-10-26 17:55:51 +0900 | [diff] [blame] | 187 | TRY_CC_MSG=echo "    CHK $(3)" 1>&2; | 
| Jiri Olsa | 28d213b | 2012-10-09 17:50:01 +0200 | [diff] [blame] | 188 |  | 
| Michael Witten | 7fbd065 | 2011-04-12 20:27:59 +0000 | [diff] [blame] | 189 | try-cc = $(shell sh -c						  \ | 
|  | 190 | 'TMP="$(OUTPUT)$(TMPOUT).$$$$";				  \ | 
| Jiri Olsa | 28d213b | 2012-10-09 17:50:01 +0200 | [diff] [blame] | 191 | $(TRY_CC_MSG)						  \ | 
| Michael Witten | 7fbd065 | 2011-04-12 20:27:59 +0000 | [diff] [blame] | 192 | echo "$(1)" |						  \ | 
| Jiri Olsa | 28d213b | 2012-10-09 17:50:01 +0200 | [diff] [blame] | 193 | $(CC) -x c - $(2) -o "$$TMP" $(TRY_CC_OUTPUT) && echo y; \ | 
| Michael Witten | 7fbd065 | 2011-04-12 20:27:59 +0000 | [diff] [blame] | 194 | rm -f "$$TMP"') |