惯性聚合 高效追踪和阅读你感兴趣的博客、新闻、科技资讯
阅读原文 在惯性聚合中打开

推荐订阅源

酷 壳 – CoolShell
酷 壳 – CoolShell
H
Hacker News: Front Page
P
Palo Alto Networks Blog
T
ThreatConnect
Apple Machine Learning Research
Apple Machine Learning Research
博客园_首页
T
True Tiger Recordings
P
Privacy & Cybersecurity Law Blog
B
Blog
IT之家
IT之家
Last Week in AI
Last Week in AI
F
Full Disclosure
Hacker News: Ask HN
Hacker News: Ask HN
C
Comments on: Blog
Microsoft Azure Blog
Microsoft Azure Blog
C
Cybersecurity and Infrastructure Security Agency CISA
Microsoft Security Blog
Microsoft Security Blog
博客园 - 【当耐特】
N
News and Events Feed by Topic
NISL@THU
NISL@THU
腾讯CDC
雷峰网
雷峰网
Security Latest
Security Latest
李成银的技术随笔
M
Microsoft Research Blog - Microsoft Research
L
LangChain Blog
L
Lohrmann on Cybersecurity
cs.CL updates on arXiv.org
cs.CL updates on arXiv.org
C
Check Point Blog
Y
Y Combinator Blog
Recent Announcements
Recent Announcements
博客园 - Franky
N
News | PayPal Newsroom
V
V2EX
A
About on SuperTechFans
The Register - Security
The Register - Security
月光博客
月光博客
奇客Solidot–传递最新科技情报
奇客Solidot–传递最新科技情报
Google Online Security Blog
Google Online Security Blog
MyScale Blog
MyScale Blog
Cisco Talos Blog
Cisco Talos Blog
Vercel News
Vercel News
WordPress大学
WordPress大学
C
Cyber Attacks, Cyber Crime and Cyber Security
The Hacker News
The Hacker News
IntelliJ IDEA : IntelliJ IDEA – the Leading IDE for Professional Development in Java and Kotlin | The JetBrains Blog
IntelliJ IDEA : IntelliJ IDEA – the Leading IDE for Professional Development in Java and Kotlin | The JetBrains Blog
爱范儿
爱范儿
A
Arctic Wolf
L
LINUX DO - 最新话题
freeCodeCamp Programming Tutorials: Python, JavaScript, Git & More

人人都是产品经理

“人货场模型”深度拆解:分析框架、建模思路、业务建议 – 人人都是产品经理, 万字干货:这可能是全网最实战的「用 Claude Code 做产品」完整方法论 – 人人都是产品经理, AI PM 的 PRD,越写越像半截草稿 – 人人都是产品经理 AI产品如何从 Skill 走到虚拟员工? – 人人都是产品经理, FDE 是什么:不是销售工程师,也不是咨询顾问 – 人人都是产品经理 建设中医科研数据库和西医科研数据库,到底差别在哪?(一) – 人人都是产品经理, 图片转 Prompt · Web Coding 工作流 – 人人都是产品经理 一文看懂VLM:自动驾驶里那个会看图说话的AI – 人人都是产品经理, 模型越强,为什么 Agent 框架反而更重要? – 人人都是产品经理, 从AI智能体演进史,看智能硬件的过去与未来 – 人人都是产品经理, 一个好的记忆Agent是什么样的? – 人人都是产品经理, AI 不再是 App,AI 是新的电脑,只是大多数人还没反应过来 – 人人都是产品经理, 小红书电商内容团队组织架构解析 – 人人都是产品经理, 给 Agent 装上眼睛和手:OpenCLI 深度体验 – 人人都是产品经理 一个AI产品经理的30天深度复盘:我发现了五个“反常识”铁律 – 人人都是产品经理, 跨境票据系统:为了开票or留证?04 – 人人都是产品经理, AI 产品经理如何设计模型路由策略 – 人人都是产品经理, 从碳纤维工厂跑出来的 AI PM:不搞套壳对话框,我靠三个“土味”项目干翻了业务痛点 – 人人都是产品经理, 家用人形机器人为坐姿用户递水 – 人人都是产品经理, 从按钮到对话:AI 时代软件形态的演进与可能 – 人人都是产品经理, 做陪伴机器人,我最先想清楚的是 AI 不该做什么 – 人人都是产品经理, 深度拆解蚂蚁阿福和氢离子:医疗AI产品的5个核心分野与10条PM启示 – 人人都是产品经理 微调 vs RAG,AI产品经理怎么选? – 人人都是产品经理, 从木鸟、途家、美团首页设计,看流量分发和业务逻辑 – 人人都是产品经理, 拆解具身智能(硬件本体):基本定义、组成部件和商业化 – 人人都是产品经理, 一文看懂 OpenHarmony 跨平台框架生态:9 大仓库全解析 – 人人都是产品经理, 黄仁勋:AI Infra 将建设十年,Agent 才刚刚开始进入生产系统 – 人人都是产品经理, 黄仁勋:AI Infra 将建设十年,Agent 才刚刚开始进入生产系统 – 人人都是产品经理, Claude 都能写高保真原型了,为什么 Anthropic 还要单独做 Claude Design – 人人都是产品经理, Claude 都能写高保真原型了,为什么 Anthropic 还要单独做 Claude Design – 人人都是产品经理, 从 Demo 到上线,AI 产品经理绕不开 Pipeline – 人人都是产品经理, 一人公司最大的坑,是什么都想自己干 – 人人都是产品经理, 从压力检测器到压力管理教练:AI 产品价值在哪里 – 人人都是产品经理, ChatGPT 对话太多,之前聊的好东西找不到了 – 人人都是产品经理, 写了五年银行数据之后,我对 AI 产品的看法 – 人人都是产品经理, 数字资产应税处理 – 人人都是产品经理, 全球用户破千万,4000万美元ARR,海外AI 3D赛道实力派都有谁? – 人人都是产品经理, 四个步骤,零基础小白也能写好Skills – 人人都是产品经理, 四个步骤,零基础小白也能写好Skills – 人人都是产品经理, 天坑专业、0经验、面试被拒N次,死磕AI产品经理2个月拿到offer! – 人人都是产品经理, 为什么企业“会做营销”,成了反义词? – 人人都是产品经理, 为什么说Windows在AI时代变成了落后生产力? – 人人都是产品经理, 从会聊天到会办事,Amazon Quick 让我重新理解办公 AI – 人人都是产品经理, 你被豆包骗了?说了,是好事! – 人人都是产品经理, Salesforce开放API AI直接调用,护城河松动?产品该怎么做? – 人人都是产品经理, 县城创业者勇闯缅甸,40天挣扎煎熬,一场被迫结束的出海梦 – 人人都是产品经理, 做B端业务,我为什么对更聪明的AI毫无期待 – 人人都是产品经理, AI陪伴为何聊着聊着就不想打开了?——从依恋理论看设计反思 – 人人都是产品经理, AI 都开始执行任务了,RAG 还要停在知识库吗? – 人人都是产品经理, 双原生架构,不用错AI的关键 微软AI掌门人放话:12到18个月后,白领工作将被AI全面接管? – 人人都是产品经理, Google I/O 2026:Flash 这次值得仔细看 – 人人都是产品经理, K12会员转化率从0.5%到3.0%——系统要素拆解法实战②:构建“零风险”信任底座 – 人人都是产品经理, 搜索已死,AI 未生:一场被算法肢解的互联网入口 – 人人都是产品经理, 90% 的 AI产品经理都在做错竞品分析(包括 4 周前的我) – 人人都是产品经理, 腾讯CodeBuddy实战手记:体系化应用开发的四个坑与我的解法 – 人人都是产品经理, DeepSeek 狂招评测工程师,我看懂了:大模型的盲盒时代彻底结束 – 人人都是产品经理, Agent = Model + Harness:理解 AI Agent 可靠性的关键概念 – 人人都是产品经理, MCP热潮过去后,产品经理该看什么 – 人人都是产品经理, 0 基础搭一个“在线健康管家”AI 客服:RAG、槽位填充、数据库怎么配合? – 人人都是产品经理, 大模型时代下,缓存命中率如何影响AI产品体验与成本 – 人人都是产品经理, AI没有杀死设计师,是设计先变无聊了 – 人人都是产品经理, 归因分析:用户成交,到底该算谁的功劳? – 人人都是产品经理, 做 AI 角色陪伴产品竞品分析时,我会重点看这 7 件事 – 人人都是产品经理, 别再迷信沉重的 AI 壳子了:从 DeepSeek-Reasonix 看 B 端真正的 ROI 刺客 – 人人都是产品经理, 人类用了2400年,终于把“思考”交给了机器 – 人人都是产品经理, 所有产品,都要被 Agent 重构一遍 – 人人都是产品经理, Anthropic 是如何做AI产品的? – 人人都是产品经理, 跨境资金系统:不是记余额,而是解释差异03 – 人人都是产品经理, Anthropic 出的这份《创始人手册》,值得每个创业者读一遍 – 人人都是产品经理 Nexus:RAG 时代终结?编译器 AI 知识层来了 – 人人都是产品经理, Nexus:RAG 时代终结?编译器 AI 知识层来了 – 人人都是产品经理, 鸿蒙 PC 平台 Rust 环境极简搭建指南|附 AI 开发神器 AtomCode 推荐 – 人人都是产品经理, 什么是OPC?有哪些成功的OPC商业模式? – 人人都是产品经理, 人与AI恋爱的“罪与罚” – 人人都是产品经理, 别再给 AI 盲目套对话框了:从 Cursor 2.0 看 B 端 AI 的形态之争 – 人人都是产品经理, 数字员工为什么会遗忘:AI记忆的底层缺陷与系统性补偿(上篇) – 人人都是产品经理 用户要的不是“导出按钮”:产品经理如何从功能诉求里识别真实需求 – 人人都是产品经理 电商公司的终局,到底是什么? – 人人都是产品经理, 不写PRD的第三周,我发现产品经理的活已经变了 挫败感:被忽视的产品杀手 – 人人都是产品经理, 微信读书 + AI = ?我试出了最惊艳的用法 拼多多强付费产品如何定价 – 人人都是产品经理, 跨境财税系统的总体架构设计02 – 人人都是产品经理, 什么叫AI原生组织?如何打造AI原生组织? – 人人都是产品经理, AI日报 · 2026年5月19日 – 人人都是产品经理 AI日报 · 2026年5月19日 – 人人都是产品经理 AI日报 · 2026年5月19日 – 人人都是产品经理 我用 AI 做完一份 PPT 和一个后台页面,才看懂这一代模型在变什么 – 人人都是产品经理, 豆包收费上热搜:从免费搭子到付费会员,我们为什么不买账? – 人人都是产品经理, 百次客户对话提炼:值得收藏的演示汇报预期管理指南 – 人人都是产品经理 Claude code版本的Vibe Design:给产品经理装上设计的AI外挂 – 人人都是产品经理, 贴贴靠谱吗?基于利益相关者理论的三维可信度分析 – 人人都是产品经理, 推荐两个小众却值钱的PM Skill – 人人都是产品经理, MCP 工作流,我是怎么走通的 – 人人都是产品经理, 一个人跑完 AI 情绪产品从 0 到 1,我沉淀了一套练手项目复盘方法论 – 人人都是产品经理 不会写代码,怎么用AI三天搞出一个能赚钱的产品 分享面试时被问到常用的面试题(上) – 人人都是产品经理, B端产品AI升级:存量报表和预警功能的智能改造经验 – 人人都是产品经理, 我跟 AI 学产品,是从一个”粉白色老头”开始的 – 人人都是产品经理
鸿蒙PC三方库构建总指挥HPKBUILD(sha)库为例 – 人人都是产品经理,
nutpi · 2026-04-17 · via 人人都是产品经理

HPKBUILD 是 OpenHarmony 三方库构建系统的核心脚本,如同 PKGBUILD 之于 Arch Linux。它仅用百余行代码,便精准定义了从源码获取、编译、安装到打包的完整自动化流程。本文将以 SHA 库为例,逐行拆解其作为“总指挥”的设计逻辑与关键函数,助你掌握为任意库编写 HPKBUILD 的能力。

如果你用过 Arch Linux,一定知道 PKGBUILD——一个脚本文件定义了包的名字、版本、怎么下载源码、怎么编译、怎么打包。Lycium 构建系统的 HPKBUILD 就是这个思路的 OpenHarmony 版本。

HPKBUILD 是整个三方库适配的总指挥。它告诉构建系统:这个包叫什么、从哪下载源码、支持哪些 CPU 架构、怎么准备源码、怎么编译、怎么安装、怎么打包。所有构建流程的决策都集中在这一个文件里。

SHA 库的 HPKBUILD 只有 106 行,但每一行都有明确的作用。这篇文章就从头到尾拆解一遍,让你看完之后能自己写一个 HPKBUILD。

项目地址:https://atomgit.com/oh-tpc/pc_sha

背景:HPKBUILD 在构建系统中的角色

Lycium 构建系统简介

Lycium 是 OpenHarmony 的三方库构建系统,负责把开源库从源码编译成可在 OpenHarmony 设备上安装的 HNP 包。它的核心工作流程是:

源码获取 → 补丁应用 → 编译构建 → 安装产物 → 打包发布

HPKBUILD 就是定义这个流程的脚本。Lycium 的主构建脚本(build.sh)读取 HPKBUILD,按顺序调用里面的函数,完成从源码到包的全过程。

HPKBUILD 的函数生命周期

┌─────────────┐

│ prepare() │ 准备:下载源码、应用补丁、创建构建目录

└──────┬──────┘

┌─────────────┐

│ build() │ 编译:运行 CMake 配置和 make 编译

└──────┬──────┘

┌─────────────┐

│ package() │ 安装:把编译产物安装到目标目录

└──────┬──────┘

┌─────────────┐

│ archive() │ 打包:生成 tar.gz 和 HNP 包

└──────┬──────┘

┌─────────────┐

│ check() │ 测试:验证功能正确性(需在设备上执行)

└─────────────┘

这五个函数就是 HPKBUILD 的骨架,每个函数负责构建流程的一个阶段。

完整文件结构总览

HPKBUILD

├── 版权声明(第 1-15 行)

├── 环境加载(第 17 行)

├── 包元数据(第 19-27 行)

├── 源码配置(第 29-35 行)

├── prepare() 函数(第 37-57 行)

├── build() 函数(第 59-66 行)

├── archive() 函数(第 67-89 行)

├── package() 函数(第 91-95 行)

├── check() 函数(第 97-100 行)

└── cleanbuild() 函数(第 103-105 行)

下面逐段拆解。

第一段:版权声明(第 1-15 行)

# Copyright (c) 2023 Huawei Device Co., Ltd.

# Licensed under the Apache License, Version 2.0 (the “License”);

# you may not use this file except in compliance with the License.

# You may obtain a copy of the License at

#

# http://www.apache.org/licenses/LICENSE-2.0

#

# Unless required by applicable law or agreed to in writing, software

# distributed under the License is distributed on an “AS IS” BASIS,

# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.

# See the License for the specific language governing permissions and

# limitations under the License.

# Contributor: huangminzhong <huangminzhong2@huawei.com>

# Maintainer: huangminzhong <huangminzhong2@huawei.com>

通俗理解:这是文件的”身份证”——谁写的、什么许可证、谁负责维护。

Apache 2.0 是 OpenHarmony 项目统一使用的许可证,要求保留版权声明和许可证文本。

第二段:环境加载(第 17 行)

source envset.sh

通俗理解:加载”工具箱”——envset.sh 里定义了各种架构的环境设置函数。

这一行看似简单,但它是解决 HNP 打包问题的关键。envset.sh 提供了以下函数:

这些函数在 archive() 阶段被调用,确保 HNP 打包工具能找到正确的工具链。如果缺少这一行,打包时会报 pack: command not found 错误。

第三段:包元数据(第 19-27 行)

pkgname=sha

pkgver=3ee0d88fc4f629b2e084f1b4cbf22cd3597542fb

pkgrel=0

pkgdesc=””

url=”https://github.com/BrianGladman/sha”

archs=(“armeabi-v7a” “arm64-v8a”)

license=(“the sha license”)

depends=()

makedepends=()

通俗理解:这是包的”名片”——名字、版本、主页、支持什么架构、依赖什么。

逐变量解读

pkgname=sha

包名。整个构建系统用这个名字来标识包——目录叫 sha/,产物叫 sha.hnp,日志叫 sha_xxx.log。

pkgver=3ee0d88fc4f629b2e084f1b4cbf22cd3597542fb

版本号。这里用的是 Git commit hash 而不是 1.0.0 这样的语义化版本,因为上游仓库没有发布正式版本号。

为什么用 commit hash?

  • 精确:唯一标识源码的某一确切状态
  • 可追溯:git show 3ee0d88 就能看到具体提交内容
  • 可重现:确保每次构建基于完全相同的源码

pkgrel=0

包释放版本。同一个 pkgver 的第几次打包。比如修复了打包脚本但源码没变,就把 pkgrel 从 0 改成 1。当前为 0,表示首次打包。

pkgdesc=“”

包描述。当前为空,建议补充为类似 “SHA hash algorithm library for OpenHarmony” 的描述。

url=”https://github.com/BrianGladman/sha”

上游仓库主页。供人浏览用,和 source 变量(用于 git clone)略有不同。

archs=(“armeabi-v7a” “arm64-v8a”)

支持的 CPU 架构列表。

构建系统会为每个架构分别执行一遍完整的构建流程,生成架构特定的产物。

license=(“the sha license”)

许可证声明。这里用的是自定义字符串,建议改为标准 SPDX 标识符如 “BSD-2-Clause”。

depends=() 和 makedepends=()

运行时依赖和构建时依赖。SHA 库是纯算法库,不依赖任何其他库,所以都是空数组。

第四段:源码配置(第 29-35 行)

source=”https://github.com/BrianGladman/sha.git”

autounpack=false

downloadpackage=false

builddir=$pkgname-${pkgver}

download_and_patch_flag=true

通俗理解:告诉构建系统”从哪拿源码”和”怎么拿”。

逐变量解读

source=”https://github.com/BrianGladman/sha.git”

源码地址。注意末尾有 .git,表示用 git clone 获取,而不是下载压缩包。

autounpack=false

不自动解压。因为用的是 git clone,不需要解压步骤。

downloadpackage=false

不下载压缩包。和 autounpack 配合使用,明确告诉构建系统“我们用 Git 管理源码”。

builddir=$pkgname-${pkgver}

构建目录名。展开后是 sha-3ee0d88fc4f629b2e084f1b4cbf22cd3597542fb。源码就放在这个目录里。

download_and_patch_flag=true

下载和补丁标志。true 表示还没下载过,prepare() 函数会执行下载和补丁操作。下载完成后设为 false,避免重复下载。

第五段:prepare() 函数(第 37-57 行)

prepare() {

if [ “$download_and_patch_flag” == true ]

then

git clone$source$builddir

if [ $? -ne 0 ]

then

return -1

fi

cd$builddir

git reset –hard $pkgver

if [ $? -ne 0 ]

then

cd$OLDPWD

return -2

fi

patch -p1 < ../sha_ohos.patch

cd$OLDPWD

download_and_patch_flag=false

fi

mkdir -p $builddir/$ARCH-build

}

通俗理解:准备阶段——下载源码、锁定版本、打补丁、创建构建目录。

执行流程图

prepare() 开始

├─ download_and_patch_flag == true ?

│ │

│ ├─ 是 → 执行下载和补丁

│ │ │

│ │ ├─

1. git clone 源码

│ │ │ 失败 → return -1

│ │ │

│ │ ├─

2. git reset –hard 锁定版本

│ │ │ 失败 → return -2

│ │ │

│ │ ├─

3. patch -p1 应用 OpenHarmony 补丁

│ │ │

│ │ └─

4. 设 download_and_patch_flag=false

│ │

│ └─ 否 → 跳过(已经下载过了)

└─ mkdir -p 创建架构构建目录

逐步解读

步骤 1:git clone

git clone $source $builddir

if [ $? -ne 0 ]

then

return -1

fi

从 GitHub 克隆源码到构建目录。$? 是上一条命令的退出码,0 表示成功,非 0 表示失败。失败时返回 -1,构建系统会中止并报告“源码下载失败”。

步骤 2:git reset –hard

cd $builddir

git reset –hard $pkgver

if [ $? -ne 0 ]

then

cd $OLDPWD

return -2

fi

把源码切换到指定版本。git reset –hard 会把工作目录重置到某个 commit 的状态,丢弃所有其他修改。这确保了无论仓库当前在什么状态,构建时用的都是同一个版本。

失败时先 cd $OLDPWD 回到原目录,再返回 -2。$OLDPWD 是 Bash 的内置变量,保存上一次的目录路径。

为什么不用 git checkout?reset –hard 更彻底——它不仅切换版本,还清理工作区,确保没有残留文件干扰构建。

步骤 3:应用补丁

patch -p1 < ../sha_ohos.patch

cd $OLDPWD

把 OpenHarmony 适配补丁应用到源码上。-p1 表示去掉路径的第一层(sha/),因为我们在源码目录内执行 patch。

步骤 4:标记完成

download_and_patch_flag=false

标记“已下载”,下次调用 prepare() 时不会重复下载。这在多架构构建时很重要——ARM32 和 ARM64 共享同一份源码,只需下载一次。

步骤 5:创建构建目录

mkdir -p $builddir/$ARCH-build

为当前架构创建独立的构建目录,比如 sha-3ee0d88…/arm64-v8a-build。-p 参数表示如果父目录不存在就自动创建,如果目录已存在也不报错。

为什么每个架构要单独的构建目录? 因为不同架构的编译选项、工具链、产物都不同,混在一起会冲突。

第六段:build() 函数(第 59-66 行)

build() {

cd $builddir

${OHOS_SDK}/native/build-tools/cmake/bin/cmake “$@” -B$ARCH-build -S./ > $buildlog 2>&1

$MAKE VERBOSE=1 -C $ARCH-build >> $buildlog 2>&1

ret=$?

cd $OLDPWD

return $ret

}

通俗理解:编译阶段——用 OpenHarmony SDK 的 CMake 配置项目,然后 make 编译。

逐步解读

CMake 配置

${OHOS_SDK}/native/build-tools/cmake/bin/cmake “$@” -B$ARCH-build -S./ > $buildlog 2>&1

为什么用 SDK 自带的 CMake? OpenHarmony SDK 的 CMake 预配置了交叉编译的工具链路径和系统根目录,能正确找到 OpenHarmony 的编译器和系统头文件。系统自带的 CMake 不知道这些信息。

Make 编译

$MAKE VERBOSE=1 -C $ARCH-build >> $buildlog 2>&1

返回结果

ret=$?

cd $OLDPWD

return $ret

捕获编译退出码,回到原目录,返回编译结果。0 表示成功,非 0 表示有编译错误。

第七段:archive() 函数(第 67-89 行)

archive() {

mkdir -p ${LYCIUM_ROOT}/output/$ARCH

pushd$LYCIUM_ROOT/usr/$pkgname/$ARCH

tar -zvcf ${LYCIUM_ROOT}/output/$ARCH/${pkgname}_${pkgver}.tar.gz *

popd

cp hnp.json $LYCIUM_ROOT/usr/$pkgname/$ARCH

# 设置架构相关的环境变量

if [ “$ARCH” == “armeabi-v7a” ]; then

setarm32ENV

elif [ “$ARCH” == “arm64-v8a” ]; then

setarm64ENV

fi

${HNP_TOOL} pack -i ${LYCIUM_ROOT}/usr/$pkgname/$ARCH -o ${LYCIUM_ROOT}/output/$ARCH/

# 清理环境变量

if [ “$ARCH” == “armeabi-v7a” ]; then

unsetarm32ENV

elif [ “$ARCH” == “arm64-v8a” ]; then

unsetarm64ENV

fi

}

通俗理解:打包阶段——把安装产物压缩成 tar.gz,再打包成 HNP 格式。

这是 HPKBUILD 中最复杂的函数,也是曾经出过 bug 的地方。逐步拆解:

步骤 1:创建输出目录

mkdir -p ${LYCIUM_ROOT}/output/$ARCH

创建最终产物的输出目录,如 output/arm64-v8a/。

步骤 2:打包 tar.gz

pushd $LYCIUM_ROOT/usr/$pkgname/$ARCH

tar -zvcf ${LYCIUM_ROOT}/output/$ARCH/${pkgname}_${pkgver}.tar.gz *

popd

进入安装目录,把所有文件打包成 sha_3ee0d88….tar.gz。pushd/popd 是 Bash 的目录栈操作,相当于“临时 cd 进去,干完活自动 cd 回来”。

步骤 3:复制 hnp.json

cp hnp.json $LYCIUM_ROOT/usr/$pkgname/$ARCH

把 hnp.json 复制到安装目录。HNP 打包工具需要读取这个文件来识别包的元数据。

步骤 4:设置架构环境变量

if [ “$ARCH” == “armeabi-v7a” ]; then

setarm32ENV

elif [ “$ARCH” == “arm64-v8a” ]; then

setarm64ENV

fi

这是关键修复点! 根据当前架构设置对应的环境变量,包括:

  • CC:C 编译器路径(如 arm-linux-ohos-clang)
  • CXX:C++ 编译器路径
  • HNP_TOOL:HNP 打包工具路径

为什么需要这一步? 因为 hnpcli(HNP 打包工具)需要知道当前架构的工具链信息。如果不设置环境变量,${HNP_TOOL} pack 会找不到 pack 命令,报错 pack: command not found。

步骤 5:打包 HNP

${HNP_TOOL} pack -i ${LYCIUM_ROOT}/usr/$pkgname/$ARCH -o ${LYCIUM_ROOT}/output/$ARCH/

执行后生成 sha.hnp 文件。

步骤 6:清理环境变量

if [ “$ARCH” == “armeabi-v7a” ]; then

unsetarm32ENV

elif [ “$ARCH” == “arm64-v8a” ]; then

unsetarm64ENV

fi

打包完成后,清理之前设置的环境变量。为什么需要清理? 因为构建系统会连续为多个架构执行构建——先 ARM32 再 ARM64。如果不清理,ARM32 的环境变量会污染 ARM64 的构建,导致用错误的编译器。

第八段:package() 函数(第 91-95 行)

package() {

cd $builddir

$MAKE VERBOSE=1 -C $ARCH-build install >> $buildlog 2>&1

cd $OLDPWD

}

通俗理解:安装阶段——执行 make install,把编译产物安装到标准目录结构。

CMake 的 install() 指令(在 CMakeLists.txt 中定义)决定了文件安装到哪:

最终安装到 ${LYCIUM_ROOT}/usr/sha/$ARCH/ 目录下。

第九段:check() 函数(第 97-100 行)

check() {

echo “The test must be on an OpenHarmony device!”

# ctest

}

通俗理解:测试阶段——但当前只是打印提示信息,没有实际执行测试。

ctest 被注释掉了,因为 SHA 库的测试需要在真实的 OpenHarmony 设备上运行。在开发机上交叉编译出的可执行文件无法直接运行。

实际的测试逻辑在 HPKCHECK 文件中定义,通过 openharmonycheck() 函数在设备上执行。

第十段:cleanbuild() 函数(第 103-105 行)

cleanbuild(){

rm -rf ${PWD}/$builddir #${PWD}/$packagename

}

通俗理解:清理构建目录,删除源码和构建产物。

rm -rf 会递归强制删除整个构建目录,包括源码、编译产物、构建日志等。下次构建时会重新 git clone 下载源码。

变量依赖关系图

HPKBUILD 中定义的变量之间有依赖关系,理解这些关系有助于整体把握:

pkgname ─────┬──→ builddir ($pkgname-${pkgver})

pkgver ──────┤

source ──────┼──→ prepare() 中的 git clone

archs ───────┼──→ 决定循环次数(每个架构执行一遍)

ARCH ────────┼──→ 由构建系统设置,决定当前构建的架构

│ ├→ $builddir/$ARCH-build (构建目录)

│ ├→ ${LYCIUM_ROOT}/usr/$pkgname/$ARCH (安装目录)

│ └→ ${LYCIUM_ROOT}/output/$ARCH (输出目录)

OHOS_SDK ────┼──→ build() 中的 CMake 路径

LYCIUM_ROOT ─┼──→ archive() 中的输出路径

HNP_TOOL ────┴──→ archive() 中的打包命令

多架构构建的完整流程

HPKBUILD 最精妙的设计是支持多架构构建。构建系统会为 archs 中的每个架构分别执行一遍完整流程:

┌─────────────────────────────────────────────────────────┐

│ 架构 1: armeabi-v7a │

│ │

│ ARCH=armeabi-v7a │

│ prepare() → 下载源码(仅首次)、创建 arm32 构建目录 │

│ build() → 用 arm-linux-ohos-clang 编译 │

│ package() → 安装到 usr/sha/armeabi-v7a/ │

│ archive() → setarm32ENV → 打包 → unsetarm32ENV │

└─────────────────────────────────────────────────────────┘

┌─────────────────────────────────────────────────────────┐

│ 架构 2: arm64-v8a │

│ │

│ ARCH=arm64-v8a │

│ prepare() → 跳过下载(已下载)、创建 arm64 构建目录 │

│ build() → 用 aarch64-linux-ohos-clang 编译 │

│ package() → 安装到 usr/sha/arm64-v8a/ │

│ archive() → setarm64ENV → 打包 → unsetarm64ENV │

└─────────────────────────────────────────────────────────┘

关键设计:

  • 源码只下载一次:download_and_patch_flag 确保不会重复 clone
  • 构建目录隔离:每个架构有独立的 $ARCH-build 目录
  • 环境变量隔离:setarmXXENV / unsetarmXXENV 确保架构间不污染

常见问题

Q1:为什么用 git clone 而不是下载压缩包?

SHA 库的上游仓库没有发布正式的 release 压缩包,只有 Git 仓库。用 git clone + git reset –hard 可以精确获取任意 commit 的源码,而压缩包通常只对应 release 版本。

Q2:prepare() 中的错误码 -1 和 -2 有什么区别?

  • return -1:git clone 失败(网络问题、仓库不存在等)
  • return -2:git reset 失败(commit hash 不存在等)

不同的错误码便于构建系统区分问题类型,给出更有针对性的错误提示。

Q3:为什么 build() 中用 $MAKE 而不是直接写 make?

$MAKE 是由构建系统设置的变量,可能是 make -j8(并行编译)或 ninja(更快的构建工具)。用变量而不是硬编码,让构建系统有灵活选择的余地。

Q4:archive() 中为什么要先 setarmXXENV 再 unsetarmXXENV?

setarmXXENV 设置了 HNP_TOOL 等环境变量,${HNP_TOOL} pack 才能正常执行。执行完后必须 unsetarmXXENV 清理,否则下一个架构的构建会被上一个架构的环境变量污染。

Q5:check() 为什么是空的?

SHA 库的测试程序是编译出的 ARM 可执行文件,无法在 x86 开发机上直接运行。测试需要在 OpenHarmony 设备上执行,由 HPKCHECK 文件定义。

Q6:如何只构建一个架构?

修改 archs 数组:

# 只构建 ARM64

archs=(“arm64-v8a”)

或者通过构建系统的命令行参数指定架构(如果支持的话)。

写一个 HPKBUILD 的模板

理解了 SHA 库的 HPKBUILD,你可以用以下模板为其他库编写 HPKBUILD:

# Copyright (c) 2023 Huawei Device Co., Ltd.

# Licensed under the Apache License, Version 2.0 (the “License”);

# …(版权声明)

# Contributor: 你的名字 <你的邮箱>

# Maintainer: 你的名字 <你的邮箱>

source envset.sh

# === 包元数据 ===

pkgname=库名

pkgver=版本号

pkgrel=0

pkgdesc=”包描述”

url=”上游仓库主页”

archs=(“armeabi-v7a””arm64-v8a”)

license=(“许可证”)

depends=() # 运行时依赖

makedepends=() # 构建时依赖

# === 源码配置 ===

source=”源码地址”

autounpack=false

downloadpackage=false

builddir=$pkgname-${pkgver}

download_and_patch_flag=true

# === 函数定义 ===

prepare() {

if [ “$download_and_patch_flag” == true ]; then

git clone$source$builddir || return -1

cd$builddir

git reset –hard $pkgver || { cd$OLDPWD; return -2; }

patch -p1 < ../补丁文件.patch

cd$OLDPWD

download_and_patch_flag=false

fi

mkdir -p $builddir/$ARCH-build

}

build() {

cd$builddir

${OHOS_SDK}/native/build-tools/cmake/bin/cmake “$@” -B$ARCH-build -S./ > $buildlog 2>&1

$MAKE VERBOSE=1 -C $ARCH-build >> $buildlog 2>&1

ret=$?

cd$OLDPWD

return$ret

}

package() {

cd$builddir

$MAKE VERBOSE=1 -C $ARCH-build install >> $buildlog 2>&1

cd$OLDPWD

}

archive() {

mkdir -p ${LYCIUM_ROOT}/output/$ARCH

pushd$LYCIUM_ROOT/usr/$pkgname/$ARCH

tar -zvcf ${LYCIUM_ROOT}/output/$ARCH/${pkgname}_${pkgver}.tar.gz *

popd

cp hnp.json $LYCIUM_ROOT/usr/$pkgname/$ARCH

if [ “$ARCH” == “armeabi-v7a” ]; then

setarm32ENV

elif [ “$ARCH” == “arm64-v8a” ]; then

setarm64ENV

fi

${HNP_TOOL} pack -i ${LYCIUM_ROOT}/usr/$pkgname/$ARCH -o ${LYCIUM_ROOT}/output/$ARCH/

if [ “$ARCH” == “armeabi-v7a” ]; then

unsetarm32ENV

elif [ “$ARCH” == “arm64-v8a” ]; then

unsetarm64ENV

fi

}

check() {

echo”The test must be on an OpenHarmony device!”

}

cleanbuild() {

rm -rf ${PWD}/$builddir

}

总结

HPKBUILD 是 SHA 库构建的“总指挥”,106 行代码定义了从源码到 HNP 包的完整流程。

核心结构

关键设计决策

  1. Git 管理源码:git clone + git reset –hard 精确控制版本
  2. 多架构隔离:独立构建目录 + 环境变量设置/清理
  3. SDK 工具链:使用 OpenHarmony SDK 自带的 CMake 和编译器
  4. HNP 打包:setarmXXENV → hnpcli pack → unsetarmXXENV 确保打包成功
  5. 源码只下载一次:download_and_patch_flag 避免多架构重复下载

一句话总结

HPKBUILD 把“从哪拿源码、怎么编译、装到哪、怎么打包”这四个问题的答案,用 Bash 脚本的形式写在一个文件里,让构建系统按图索骥,自动完成从源码到 HNP 包的全过程。

本文由人人都是产品经理作者【nutpi】,微信公众号:【nutpi】,原创/授权 发布于人人都是产品经理,未经许可,禁止转载。

题图来自Unsplash,基于 CC0 协议