|  | RACK_DIR ?= .
VERSION_MAJOR := 2
VERSION := 2.git.$(shell git rev-parse --short HEAD)
# VERSION := 2.0.0
FLAGS += -Iinclude -Idep/include
include arch.mk
# Sources and build flags
SOURCES += dep/nanovg/src/nanovg.c
SOURCES += dep/osdialog/osdialog.c
SOURCES += dep/oui-blendish/blendish.c
SOURCES += dep/pffft/pffft.c dep/pffft/fftpack.c
SOURCES += $(wildcard src/*.c src/*/*.c)
SOURCES += $(wildcard src/*.cpp src/*/*.cpp)
build/src/common.cpp.o: FLAGS += -D_APP_VERSION=$(VERSION)
STANDALONE_SOURCES += adapters/standalone.cpp
FLAGS += -fPIC
LDFLAGS += -shared
ifdef ARCH_LIN
	SED := sed -i
	TARGET := libRack.so
	SOURCES += dep/osdialog/osdialog_gtk3.c
build/dep/osdialog/osdialog_gtk3.c.o: FLAGS += $(shell pkg-config --cflags gtk+-3.0)
	# This prevents static variables in the DSO (dynamic shared object) from being preserved after dlclose().
	# I don't really understand the side effects (see GCC manual), but so far tests are positive.
	FLAGS += -fno-gnu-unique
	LDFLAGS += -Wl,--whole-archive
	LDFLAGS += -static-libstdc++ -static-libgcc
	LDFLAGS += dep/lib/libGLEW.a dep/lib/libglfw3.a dep/lib/libjansson.a dep/lib/libcurl.a dep/lib/libssl.a dep/lib/libcrypto.a dep/lib/libarchive.a dep/lib/libzstd.a dep/lib/libspeexdsp.a dep/lib/libsamplerate.a dep/lib/librtmidi.a dep/lib/librtaudio.a
	LDFLAGS += -Wl,--no-whole-archive
	LDFLAGS += -lpthread -lGL -ldl -lX11 -lasound -ljack -lpulse -lpulse-simple
	LDFLAGS += $(shell pkg-config --libs gtk+-3.0)
	STANDALONE_TARGET := Rack
	STANDALONE_LDFLAGS += -static-libstdc++ -static-libgcc
	STANDALONE_LDFLAGS += -Wl,-rpath=.
endif
ifdef ARCH_MAC
	SED := sed -i ''
	TARGET := libRack.dylib
	SOURCES += dep/osdialog/osdialog_mac.m
	LDFLAGS += -lpthread -ldl
	LDFLAGS += -framework Cocoa -framework OpenGL -framework IOKit -framework CoreVideo -framework CoreAudio -framework CoreMIDI
	LDFLAGS += -Wl,-all_load
	LDFLAGS += dep/lib/libGLEW.a dep/lib/libglfw3.a dep/lib/libjansson.a dep/lib/libcurl.a dep/lib/libssl.a dep/lib/libcrypto.a dep/lib/libarchive.a dep/lib/libzstd.a dep/lib/libspeexdsp.a dep/lib/libsamplerate.a dep/lib/librtmidi.a dep/lib/librtaudio.a
	STANDALONE_TARGET := Rack
	STANDALONE_LDFLAGS += -stdlib=libc++
	# For LuaJIT to work inside plugins
	STANDALONE_LDFLAGS += -Wl,-pagezero_size,10000 -Wl,-image_base,100000000
endif
ifdef ARCH_WIN
	SED := sed -i
	TARGET := libRack.dll
	SOURCES += dep/osdialog/osdialog_win.c
	LDFLAGS += -municode
	LDFLAGS += -Wl,--export-all-symbols
	LDFLAGS += -Wl,--out-implib,$(TARGET).a
	LDFLAGS += -Wl,-Bstatic -Wl,--whole-archive
	LDFLAGS += dep/lib/libglew32.a dep/lib/libglfw3.a dep/lib/libjansson.a dep/lib/libspeexdsp.a dep/lib/libsamplerate.a dep/lib/libarchive_static.a dep/lib/libzstd.a dep/lib/libcurl.a dep/lib/libssl.a dep/lib/libcrypto.a dep/lib/librtaudio.a dep/lib/librtmidi.a
	LDFLAGS += -Wl,-Bdynamic -Wl,--no-whole-archive
	LDFLAGS += -lpthread -lopengl32 -lgdi32 -lws2_32 -lcomdlg32 -lole32 -ldsound -lwinmm -lksuser -lshlwapi -lmfplat -lmfuuid -lwmcodecdspuuid -ldbghelp
	STANDALONE_TARGET := Rack.exe
	STANDALONE_LDFLAGS += -mwindows
	STANDALONE_OBJECTS += build/Rack.res
endif
include compile.mk
# Standalone adapter
ifdef ARCH_MAC
	STANDALONE_LDFLAGS += $(MAC_SDK_FLAGS)
endif
STANDALONE_OBJECTS += $(TARGET)
-include $(STANDALONE_DEPENDENCIES)
$(STANDALONE_TARGET): $(STANDALONE_SOURCES) $(STANDALONE_OBJECTS)
	$(CXX) $(CXXFLAGS) -o $@ $^ $(STANDALONE_LDFLAGS)
# Convenience targets
all: $(TARGET) $(STANDALONE_TARGET)
dep:
	$(MAKE) -C dep
run: $(STANDALONE_TARGET)
	./$< -d
runr: $(STANDALONE_TARGET)
	./$<
debug: $(STANDALONE_TARGET)
ifdef ARCH_MAC
	lldb -- ./$< -d
endif
ifdef ARCH_WIN
	gdb --args ./$< -d
endif
ifdef ARCH_LIN
	gdb --args ./$< -d
endif
screenshot: $(STANDALONE_TARGET)
	./$< -d -t 2
perf: $(STANDALONE_TARGET)
	# Requires perf
	perf record --call-graph dwarf -o perf.data ./$< -d
	# Analyze with hotspot (https://github.com/KDAB/hotspot) for example
	hotspot perf.data
	rm perf.data
valgrind: $(STANDALONE_TARGET)
	# --gen-suppressions=yes
	# --leak-check=full
	valgrind --suppressions=valgrind.supp ./$< -d
clean:
	rm -rfv build dist *.a Rack.res *.d $(TARGET) $(STANDALONE_TARGET)
# For Windows resources
build/%.res: %.rc
ifdef ARCH_WIN
	windres $^ -O coff -o $@
endif
DIST_RES := res cacert.pem Core.json template.vcv LICENSE-GPLv3.txt
DIST_NAME := Rack-"$(VERSION)"-$(ARCH_OS_NAME)
DIST_SDK_DIR := Rack-SDK
DIST_SDK := Rack-SDK-"$(VERSION)"-$(ARCH_OS_NAME).zip
ifdef ARCH_MAC
	DIST_BUNDLE := VCV Rack $(VERSION_MAJOR).app
else
	DIST_DIR := Rack$(VERSION_MAJOR)
endif
DIST_MD := $(wildcard *.md)
DIST_HTML := $(patsubst %.md, build/%.html, $(DIST_MD))
# Target not supported for public use
dist: $(TARGET) $(STANDALONE_TARGET) $(DIST_HTML)
	rm -rf dist
	mkdir -p dist
	# Copy Rack to dist
ifdef ARCH_LIN
	mkdir -p dist/"$(DIST_DIR)"
	cp $(TARGET) dist/"$(DIST_DIR)"/
	cp $(STANDALONE_TARGET) dist/"$(DIST_DIR)"/
	$(STRIP) -s dist/"$(DIST_DIR)"/$(TARGET)
	$(STRIP) -s dist/"$(DIST_DIR)"/$(STANDALONE_TARGET)
	# Manually check that no nonstandard shared libraries are linked
	ldd dist/"$(DIST_DIR)"/$(TARGET)
	ldd dist/"$(DIST_DIR)"/$(STANDALONE_TARGET)
	# Copy resources
	cp -R $(DIST_RES) dist/"$(DIST_DIR)"/
	cp $(DIST_HTML) dist/"$(DIST_DIR)"/
	cp plugins/Fundamental/dist/Fundamental-*.vcvplugin dist/"$(DIST_DIR)"/Fundamental.vcvplugin
	# Make ZIP
	cd dist && zip -q -9 -r "$(DIST_NAME)".zip "$(DIST_DIR)"
endif
ifdef ARCH_MAC
	mkdir -p dist/"$(DIST_BUNDLE)"
	mkdir -p dist/"$(DIST_BUNDLE)"/Contents
	mkdir -p dist/"$(DIST_BUNDLE)"/Contents/Resources
	mkdir -p dist/"$(DIST_BUNDLE)"/Contents/MacOS
	cp $(TARGET) dist/"$(DIST_BUNDLE)"/Contents/Resources/
	cp $(STANDALONE_TARGET) dist/"$(DIST_BUNDLE)"/Contents/MacOS/
	$(STRIP) -S dist/"$(DIST_BUNDLE)"/Contents/Resources/$(TARGET)
	$(STRIP) -S dist/"$(DIST_BUNDLE)"/Contents/MacOS/$(STANDALONE_TARGET)
	install_name_tool -change $(TARGET) @executable_path/../Resources/$(TARGET) dist/"$(DIST_BUNDLE)"/Contents/MacOS/$(STANDALONE_TARGET)
	# Manually check that no nonstandard shared libraries are linked
	otool -L dist/"$(DIST_BUNDLE)"/Contents/Resources/$(TARGET)
	otool -L dist/"$(DIST_BUNDLE)"/Contents/MacOS/$(STANDALONE_TARGET)
	# Copy resources
	cp Info.plist dist/"$(DIST_BUNDLE)"/Contents/
	$(SED) 's/{VERSION}/$(VERSION)/g' dist/"$(DIST_BUNDLE)"/Contents/Info.plist
	cp -R $(DIST_RES) dist/"$(DIST_BUNDLE)"/Contents/Resources/
	cp $(DIST_HTML) dist/"$(DIST_BUNDLE)"/Contents/Resources/
	cp -R icon.icns dist/"$(DIST_BUNDLE)"/Contents/Resources/
	cp plugins/Fundamental/dist/Fundamental-*.vcvplugin dist/"$(DIST_BUNDLE)"/Contents/Resources/Fundamental.vcvplugin
	# Clean up and sign bundle
	xattr -cr dist/"$(DIST_BUNDLE)"
	codesign --verbose --sign "Developer ID Application: Andrew Belt (VRF26934X5)" --options runtime --entitlements Entitlements.plist --timestamp --deep dist/"$(DIST_BUNDLE)"/Contents/Resources/$(TARGET) dist/"$(DIST_BUNDLE)"
	codesign --verify --deep --strict --verbose=2 dist/"$(DIST_BUNDLE)"
	# Make standalone PKG
	mkdir -p dist/Component
	cp -R dist/"$(DIST_BUNDLE)" dist/Component/
	pkgbuild --identifier com.vcvrack.rack --component-plist Component.plist --root dist/Component --install-location /Applications dist/Component.pkg
	# Make PKG
	productbuild --distribution Distribution.xml --package-path dist dist/"$(DIST_NAME)".pkg
	productsign --sign "Developer ID Installer: Andrew Belt (V8SW9J626X)" dist/"$(DIST_NAME)".pkg dist/"$(DIST_NAME)"-signed.pkg
	mv dist/"$(DIST_NAME)"-signed.pkg dist/"$(DIST_NAME)".pkg
endif
ifdef ARCH_WIN
	mkdir -p dist/"$(DIST_DIR)"
	cp $(TARGET) dist/"$(DIST_DIR)"/
	cp $(STANDALONE_TARGET) dist/"$(DIST_DIR)"/
	$(STRIP) -s dist/"$(DIST_DIR)"/$(TARGET)
	$(STRIP) -s dist/"$(DIST_DIR)"/$(STANDALONE_TARGET)
	# Copy resources
	cp -R $(DIST_RES) dist/"$(DIST_DIR)"/
	cp $(DIST_HTML) dist/"$(DIST_DIR)"/
	cp /mingw64/bin/libwinpthread-1.dll dist/"$(DIST_DIR)"/
	cp /mingw64/bin/libstdc++-6.dll dist/"$(DIST_DIR)"/
	cp /mingw64/bin/libgcc_s_seh-1.dll dist/"$(DIST_DIR)"/
	cp plugins/Fundamental/dist/Fundamental-*.vcvplugin dist/"$(DIST_DIR)"/Fundamental.vcvplugin
	# Make ZIP
	cd dist && zip -q -9 -r "$(DIST_NAME)".zip "$(DIST_DIR)"
	# Make NSIS installer
	# pacman -S mingw-w64-x86_64-nsis
	makensis -DVERSION_MAJOR="$(VERSION_MAJOR)" -DVERSION="$(VERSION)" "-XOutFile dist/$(DIST_NAME).exe" installer.nsi
endif
	# Build Rack SDK
	mkdir -p dist/"$(DIST_SDK_DIR)"
	cp -R LICENSE* *.mk include helper.py dist/"$(DIST_SDK_DIR)"/
	mkdir -p dist/"$(DIST_SDK_DIR)"/dep/
	cp -R dep/include dist/"$(DIST_SDK_DIR)"/dep/
ifdef ARCH_LIN
	cp $(TARGET) dist/"$(DIST_SDK_DIR)"/
	$(STRIP) -s dist/"$(DIST_SDK_DIR)"/$(TARGET)
endif
ifdef ARCH_MAC
	cp $(TARGET) dist/"$(DIST_SDK_DIR)"/
	$(STRIP) -S dist/"$(DIST_SDK_DIR)"/$(TARGET)
endif
ifdef ARCH_WIN
	cp libRack.dll.a dist/"$(DIST_SDK_DIR)"/
endif
	cd dist && zip -q -9 -r "$(DIST_SDK)" "$(DIST_SDK_DIR)"
# Target not supported for public use
notarize:
ifdef ARCH_MAC
	xcrun altool --notarize-app --primary-bundle-id "com.vcvrack.rack" --username "andrew@vcvrack.com" --password @keychain:notarize --output-format xml --file dist/"$(DIST_NAME)".pkg > dist/UploadInfo.plist
	# Wait for Apple's servers to approve the app
	while true; do \
		echo "Waiting on Apple servers..." ; \
		sleep 10 ; \
		xcrun altool --notarization-info `/usr/libexec/PlistBuddy -c "Print :notarization-upload:RequestUUID" dist/UploadInfo.plist` -u "andrew@vcvrack.com" -p @keychain:notarize --output-format xml > dist/RequestInfo.plist ; \
		if [ "`/usr/libexec/PlistBuddy -c "Print :notarization-info:Status" dist/RequestInfo.plist`" != "in progress" ]; then \
			break ; \
		fi ; \
	done
	# Mark app as notarized
	xcrun stapler staple dist/"$(DIST_NAME)".pkg
	# Check notarization
	stapler validate --verbose dist/"$(DIST_NAME)".pkg
	spctl --assess --type execute --ignore-cache --no-cache -vv dist/"$(DIST_BUNDLE)"
endif
cleandist:
	rm -rfv dist
# Plugin helpers
plugins:
ifdef CMD
	for f in plugins/*; do (cd "$$f" && $(CMD)); done
else
	for f in plugins/*; do $(MAKE) -C "$$f"; done
endif
# Includes
.DEFAULT_GOAL := all
.PHONY: all dep run debug clean dist upload src plugins
 |