c_src.mk 4.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208
  1. # Copyright (c) 2014-2015, Loïc Hoguin <essen@ninenines.eu>
  2. # This file is part of erlang.mk and subject to the terms of the ISC License.
  3. .PHONY: clean-c_src distclean-c_src-env
  4. # Configuration.
  5. C_SRC_DIR ?= $(CURDIR)/c_src
  6. C_SRC_ENV ?= $(C_SRC_DIR)/env.mk
  7. C_SRC_OUTPUT ?= $(CURDIR)/priv/$(PROJECT).so
  8. C_SRC_TYPE ?= shared
  9. # System type and C compiler/flags.
  10. ifeq ($(PLATFORM),darwin)
  11. CC ?= cc
  12. CFLAGS ?= -O3 -std=c99 -arch x86_64 -finline-functions -Wall -Wmissing-prototypes
  13. CXXFLAGS ?= -O3 -arch x86_64 -finline-functions -Wall
  14. LDFLAGS ?= -arch x86_64 -flat_namespace -undefined suppress
  15. else ifeq ($(PLATFORM),freebsd)
  16. CC ?= cc
  17. CFLAGS ?= -O3 -std=c99 -finline-functions -Wall -Wmissing-prototypes
  18. CXXFLAGS ?= -O3 -finline-functions -Wall
  19. else ifeq ($(PLATFORM),linux)
  20. CC ?= gcc
  21. CFLAGS ?= -O3 -std=c99 -finline-functions -Wall -Wmissing-prototypes
  22. CXXFLAGS ?= -O3 -finline-functions -Wall
  23. endif
  24. CFLAGS += -fPIC -I $(ERTS_INCLUDE_DIR) -I $(ERL_INTERFACE_INCLUDE_DIR)
  25. CXXFLAGS += -fPIC -I $(ERTS_INCLUDE_DIR) -I $(ERL_INTERFACE_INCLUDE_DIR)
  26. LDLIBS += -L $(ERL_INTERFACE_LIB_DIR) -lerl_interface -lei
  27. # Verbosity.
  28. c_verbose_0 = @echo " C " $(?F);
  29. c_verbose = $(c_verbose_$(V))
  30. cpp_verbose_0 = @echo " CPP " $(?F);
  31. cpp_verbose = $(cpp_verbose_$(V))
  32. link_verbose_0 = @echo " LD " $(@F);
  33. link_verbose = $(link_verbose_$(V))
  34. # Targets.
  35. ifeq ($(wildcard $(C_SRC_DIR)),)
  36. else ifneq ($(wildcard $(C_SRC_DIR)/Makefile),)
  37. app:: app-c_src
  38. test-build:: app-c_src
  39. app-c_src:
  40. $(MAKE) -C $(C_SRC_DIR)
  41. clean::
  42. $(MAKE) -C $(C_SRC_DIR) clean
  43. else
  44. ifeq ($(SOURCES),)
  45. SOURCES := $(sort $(foreach pat,*.c *.C *.cc *.cpp,$(call core_find,$(C_SRC_DIR)/,$(pat))))
  46. endif
  47. OBJECTS = $(addsuffix .o, $(basename $(SOURCES)))
  48. COMPILE_C = $(c_verbose) $(CC) $(CFLAGS) $(CPPFLAGS) -c
  49. COMPILE_CPP = $(cpp_verbose) $(CXX) $(CXXFLAGS) $(CPPFLAGS) -c
  50. app:: $(C_SRC_ENV) $(C_SRC_OUTPUT)
  51. test-build:: $(C_SRC_ENV) $(C_SRC_OUTPUT)
  52. $(C_SRC_OUTPUT): $(OBJECTS)
  53. $(verbose) mkdir -p priv/
  54. $(link_verbose) $(CC) $(OBJECTS) \
  55. $(LDFLAGS) $(if $(filter $(C_SRC_TYPE),shared),-shared) $(LDLIBS) \
  56. -o $(C_SRC_OUTPUT)
  57. %.o: %.c
  58. $(COMPILE_C) $(OUTPUT_OPTION) $<
  59. %.o: %.cc
  60. $(COMPILE_CPP) $(OUTPUT_OPTION) $<
  61. %.o: %.C
  62. $(COMPILE_CPP) $(OUTPUT_OPTION) $<
  63. %.o: %.cpp
  64. $(COMPILE_CPP) $(OUTPUT_OPTION) $<
  65. clean:: clean-c_src
  66. clean-c_src:
  67. $(gen_verbose) rm -f $(C_SRC_OUTPUT) $(OBJECTS)
  68. endif
  69. ifneq ($(wildcard $(C_SRC_DIR)),)
  70. $(C_SRC_ENV):
  71. $(verbose) $(ERL) -eval "file:write_file(\"$(C_SRC_ENV)\", \
  72. io_lib:format( \
  73. \"ERTS_INCLUDE_DIR ?= ~s/erts-~s/include/~n\" \
  74. \"ERL_INTERFACE_INCLUDE_DIR ?= ~s~n\" \
  75. \"ERL_INTERFACE_LIB_DIR ?= ~s~n\", \
  76. [code:root_dir(), erlang:system_info(version), \
  77. code:lib_dir(erl_interface, include), \
  78. code:lib_dir(erl_interface, lib)])), \
  79. halt()."
  80. distclean:: distclean-c_src-env
  81. distclean-c_src-env:
  82. $(gen_verbose) rm -f $(C_SRC_ENV)
  83. -include $(C_SRC_ENV)
  84. endif
  85. # Templates.
  86. define bs_c_nif
  87. #include "erl_nif.h"
  88. static int loads = 0;
  89. static int load(ErlNifEnv* env, void** priv_data, ERL_NIF_TERM load_info)
  90. {
  91. /* Initialize private data. */
  92. *priv_data = NULL;
  93. loads++;
  94. return 0;
  95. }
  96. static int upgrade(ErlNifEnv* env, void** priv_data, void** old_priv_data, ERL_NIF_TERM load_info)
  97. {
  98. /* Convert the private data to the new version. */
  99. *priv_data = *old_priv_data;
  100. loads++;
  101. return 0;
  102. }
  103. static void unload(ErlNifEnv* env, void* priv_data)
  104. {
  105. if (loads == 1) {
  106. /* Destroy the private data. */
  107. }
  108. loads--;
  109. }
  110. static ERL_NIF_TERM hello(ErlNifEnv* env, int argc, const ERL_NIF_TERM argv[])
  111. {
  112. if (enif_is_atom(env, argv[0])) {
  113. return enif_make_tuple2(env,
  114. enif_make_atom(env, "hello"),
  115. argv[0]);
  116. }
  117. return enif_make_tuple2(env,
  118. enif_make_atom(env, "error"),
  119. enif_make_atom(env, "badarg"));
  120. }
  121. static ErlNifFunc nif_funcs[] = {
  122. {"hello", 1, hello}
  123. };
  124. ERL_NIF_INIT($n, nif_funcs, load, NULL, upgrade, unload)
  125. endef
  126. define bs_erl_nif
  127. -module($n).
  128. -export([hello/1]).
  129. -on_load(on_load/0).
  130. on_load() ->
  131. PrivDir = case code:priv_dir(?MODULE) of
  132. {error, _} ->
  133. AppPath = filename:dirname(filename:dirname(code:which(?MODULE))),
  134. filename:join(AppPath, "priv");
  135. Path ->
  136. Path
  137. end,
  138. erlang:load_nif(filename:join(PrivDir, atom_to_list(?MODULE)), 0).
  139. hello(_) ->
  140. erlang:nif_error({not_loaded, ?MODULE}).
  141. endef
  142. new-nif:
  143. ifneq ($(wildcard $(C_SRC_DIR)/$n.c),)
  144. $(error Error: $(C_SRC_DIR)/$n.c already exists)
  145. endif
  146. ifneq ($(wildcard src/$n.erl),)
  147. $(error Error: src/$n.erl already exists)
  148. endif
  149. ifdef in
  150. $(verbose) $(MAKE) -C $(APPS_DIR)/$(in)/ new-nif n=$n in=
  151. else
  152. $(verbose) mkdir -p $(C_SRC_DIR) src/
  153. $(call render_template,bs_c_nif,$(C_SRC_DIR)/$n.c)
  154. $(call render_template,bs_erl_nif,src/$n.erl)
  155. endif