GOFLAGS := -tags load_wgnt_from_rsrc -ldflags="-H windowsgui -s -w" -trimpath -buildvcs=false -v
export GOOS := windows
export PATH := $(CURDIR)/.deps/go/bin:$(PATH)
VERSION := $(shell sed -n 's/^\s*Number\s*=\s*"\([0-9.]\+\)"$$/\1/p' version/version.go)
empty :=
space := $(empty) $(empty)
comma := ,
RCFLAGS := -DWIREGUARD_VERSION_ARRAY=$(subst $(space),$(comma),$(wordlist 1,4,$(subst .,$(space),$(VERSION)) 0 0 0 0)) -DWIREGUARD_VERSION_STR=$(VERSION) -O coff -c 65001
rwildcard=$(foreach d,$(filter-out .deps,$(wildcard $1*)),$(call rwildcard,$d/,$2) $(filter $(subst *,%,$2),$d))
SOURCE_FILES := $(call rwildcard,,*.go) .deps/go/prepared go.mod go.sum
RESOURCE_FILES := resources.rc version/version.go manifest.xml $(patsubst %.svg,%.ico,$(wildcard ui/icon/*.svg)) .deps/wireguard-nt/prepared
DEPLOYMENT_HOST ?= winvm
DEPLOYMENT_PATH ?= Desktop
all: amd64/wireguard.exe x86/wireguard.exe arm64/wireguard.exe
define download =
.distfiles/$(1):
mkdir -p .distfiles
if ! curl -L#o $$@.unverified $(2); then rm -f $$@.unverified; exit 1; fi
if ! echo "$(3) $$@.unverified" | sha256sum -c; then rm -f $$@.unverified; exit 1; fi
if ! mv $$@.unverified $$@; then rm -f $$@.unverified; exit 1; fi
endef
$(eval $(call download,go.tar.gz,https://go.dev/dl/go1.18.linux-amd64.tar.gz,e85278e98f57cdb150fe8409e6e5df5343ecb13cebf03a5d5ff12bd55a80264f))
$(eval $(call download,wireguard-nt.zip,https://download.wireguard.com/wireguard-nt/wireguard-nt-0.10.1.zip,772c0b1463d8d2212716f43f06f4594d880dea4f735165bd68e388fc41b81605))
.deps/go/prepared: .distfiles/go.tar.gz
mkdir -p .deps
rm -rf .deps/go
bsdtar -C .deps -xf .distfiles/go.tar.gz
chmod -R +w .deps/go
touch $@
.deps/wireguard-nt/prepared: .distfiles/wireguard-nt.zip
mkdir -p .deps
rm -rf .deps/wireguard-nt
bsdtar -C .deps -xf .distfiles/wireguard-nt.zip
touch $@
%.ico: %.svg
convert -background none $< -define icon:auto-resize="256,192,128,96,64,48,40,32,24,20,16" -compress zip $@
resources_amd64.syso: $(RESOURCE_FILES)
x86_64-w64-mingw32-windres $(RCFLAGS) -I .deps/wireguard-nt/bin/amd64 -i $< -o $@
resources_386.syso: $(RESOURCE_FILES)
i686-w64-mingw32-windres $(RCFLAGS) -I .deps/wireguard-nt/bin/x86 -i $< -o $@
resources_arm64.syso: $(RESOURCE_FILES)
aarch64-w64-mingw32-windres $(RCFLAGS) -I .deps/wireguard-nt/bin/arm64 -i $< -o $@
amd64/wireguard.exe: export GOARCH := amd64
amd64/wireguard.exe: resources_amd64.syso $(SOURCE_FILES)
go build $(GOFLAGS) -o $@
x86/wireguard.exe: export GOARCH := 386
x86/wireguard.exe: resources_386.syso $(SOURCE_FILES)
go build $(GOFLAGS) -o $@
arm64/wireguard.exe: export GOARCH := arm64
arm64/wireguard.exe: resources_arm64.syso $(SOURCE_FILES)
go build $(GOFLAGS) -o $@
remaster: export GOARCH := amd64
remaster: export GOPROXY := direct
remaster: .deps/go/prepared
rm -f go.sum go.mod
cp go.mod.master go.mod
go get -d
sed -i $(shell curl -L 'https://go.dev/dl/?mode=json&include=all' | jq -r '(".windows-amd64.zip",".linux-amd64.tar.gz") as $$suffix | .[0].files[] | select(.filename|endswith($$suffix)) | ("-e", "s/go[0-9][^ ]*\\\($$suffix)\\([ ,]\\)[a-f0-9]\\+/\(.filename)\\1\(.sha256)/") | @sh') Makefile build.bat
fmt: export GOARCH := amd64
fmt: .deps/go/prepared
go fmt ./...
generate: export GOOS :=
generate: .deps/go/prepared
go generate -mod=mod ./...
crowdin:
find locales -maxdepth 1 -mindepth 1 -type d \! -name en -exec rm -rf {} +
curl -Lo - https://crowdin.com/backend/download/project/wireguard.zip | bsdtar -C locales -x -f - --strip-components 2 wireguard-windows
find locales -name messages.gotext.json -exec bash -c '[[ $$(jq ".messages | length" {}) -ne 0 ]] || rm -rf "$$(dirname {})"' \;
@$(MAKE) --no-print-directory generate
deploy: amd64/wireguard.exe
-ssh $(DEPLOYMENT_HOST) -- 'taskkill /im wireguard.exe /f'
scp $< $(DEPLOYMENT_HOST):$(DEPLOYMENT_PATH)
clean:
rm -rf *.syso ui/icon/*.ico x86/ amd64/ arm64/ .deps
distclean: clean
rm -rf .distfiles
.PHONY: deploy clean distclean fmt remaster generate all
这段代码是一个 Makefile,用于构建 WireGuard 的 Windows 版本。它包含了一系列的规则和变量,用于自动化构建、打包、生成资源文件和部署等任务。下面是对各个部分的详细分析:
1. 全局变量
GOFLAGS
: 设定了 Go 编译器的标志。包括:-tags load_wgnt_from_rsrc
: 启用特定的编译标签load_wgnt_from_rsrc
。-ldflags="-H windowsgui -s -w"
: 指定了 Go 链接器的标志。-H windowsgui
生成一个没有控制台窗口的 GUI 应用程序;-s -w
删除调试信息以减少可执行文件的大小。-trimpath
: 去除编译时生成的路径信息。-buildvcs=false
: 不嵌入版本控制系统信息。-v
: 让编译过程更加详细。
GOOS
: 设定目标操作系统为 Windows。PATH
: 更新 PATH 变量,将当前目录的.deps/go/bin
目录添加到 PATH 中,确保使用正确版本的 Go 工具链。VERSION
: 从version/version.go
文件中提取版本号。RCFLAGS
: 设置资源编译器的标志,用于生成.syso
文件。它将版本号转换为特定格式,用于嵌入资源文件中。
2. 函数和宏
rwildcard
: 递归地搜索目录,查找符合特定模式的文件。用于获取所有 Go 源文件和资源文件。download
: 这个宏用于从指定 URL 下载文件,验证其 SHA256 校验和,并将其存储在.distfiles
目录中。
3. 依赖文件
$(eval $(call download,...))
: 调用download
宏,下载并验证指定的文件。这里下载了 Go 工具链和wireguard-nt
的预编译二进制文件。.deps/go/prepared
和.deps/wireguard-nt/prepared
: 表示依赖文件准备完毕的标志。解压下载的文件并设置适当的权限。
4. 构建规则
resources_amd64.syso
,resources_386.syso
,resources_arm64.syso
: 使用windres
工具将资源文件编译成.syso
文件,这些文件会被链接到最终的可执行文件中。amd64/wireguard.exe
,x86/wireguard.exe
,arm64/wireguard.exe
: 这些是目标可执行文件的构建规则,分别用于 x86_64、x86 和 ARM64 架构。通过设定GOARCH
变量来确定生成的架构,并使用go build
进行编译。
5. 辅助任务
remaster
: 用于更新go.mod
文件并获取所有的依赖包。fmt
: 格式化 Go 代码。generate
: 运行go generate
来自动生成代码。crowdin
: 从 Crowdin 下载翻译文件,并更新本地化资源。deploy
: 将编译好的可执行文件部署到指定的 Windows 虚拟机上。clean
和distclean
: 用于清理构建生成的文件。
6. 总结
- 这个 Makefile 是一个完整的构建系统,涵盖了从下载依赖、生成资源、编译代码到部署的所有步骤。
- 通过设置不同的目标架构(如
amd64
,x86
,arm64
),它能够生成适用于不同平台的 WireGuard Windows 可执行文件。 - 该文件非常灵活,可以通过修改全局变量和环境变量来调整构建过程的行为。