aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authorMukilan Thiyagarajan <mukilan@igalia.com>2024-09-20 13:50:27 +0530
committerGitHub <noreply@github.com>2024-09-20 08:20:27 +0000
commit157e28c59b34ec8c7334161fda658cbbb1e66a25 (patch)
treea4f2b9b57f1c3b30333a97fb3831973c110fad99
parent457d37d94ee6966cad377c373d333a00c637e1ae (diff)
downloadservo-157e28c59b34ec8c7334161fda658cbbb1e66a25.tar.gz
servo-157e28c59b34ec8c7334161fda658cbbb1e66a25.zip
openharmony: add servoshell for ohos (#33295)
* openharmony: add servoshell for ohos Signed-off-by: Mukilan Thiyagarajan <mukilan@igalia.com> * ohos: handle missing signing config on forks Signed-off-by: Mukilan Thiyagarajan <mukilan@igalia.com> --------- Signed-off-by: Mukilan Thiyagarajan <mukilan@igalia.com>
-rw-r--r--.gitattributes1
-rw-r--r--.github/workflows/main.yml1
-rw-r--r--.github/workflows/ohos.yml35
-rw-r--r--python/servo/command_base.py33
-rw-r--r--python/servo/package_commands.py63
-rw-r--r--python/servo/platform/build_target.py25
-rw-r--r--support/openharmony/AppScope/app.json510
-rw-r--r--support/openharmony/AppScope/resources/base/element/string.json8
l---------support/openharmony/AppScope/resources/base/media/servo_64.png1
-rw-r--r--support/openharmony/build-profile.json544
-rw-r--r--support/openharmony/entry/build-profile.json536
-rw-r--r--support/openharmony/entry/hvigorfile.ts6
-rw-r--r--support/openharmony/entry/obfuscation-rules.txt18
-rw-r--r--support/openharmony/entry/oh-package.json512
-rw-r--r--support/openharmony/entry/src/main/cpp/CMakeLists.txt3
-rw-r--r--support/openharmony/entry/src/main/cpp/types/libentry/index.d.ts1
-rw-r--r--support/openharmony/entry/src/main/cpp/types/libentry/oh-package.json56
-rw-r--r--support/openharmony/entry/src/main/ets/entryability/EntryAbility.ets43
-rw-r--r--support/openharmony/entry/src/main/ets/pages/Index.ets85
-rw-r--r--support/openharmony/entry/src/main/module.json552
-rw-r--r--support/openharmony/entry/src/main/resources/base/element/color.json8
-rw-r--r--support/openharmony/entry/src/main/resources/base/element/string.json20
l---------support/openharmony/entry/src/main/resources/base/media/servo_1024.png1
l---------support/openharmony/entry/src/main/resources/base/media/servo_64.png1
-rw-r--r--support/openharmony/entry/src/main/resources/base/profile/main_pages.json5
-rw-r--r--support/openharmony/entry/src/main/resources/en_US/element/string.json16
-rw-r--r--support/openharmony/entry/src/main/resources/zh_CN/element/string.json16
-rw-r--r--support/openharmony/hvigor/hvigor-config.json517
-rw-r--r--support/openharmony/hvigorfile.ts34
-rw-r--r--support/openharmony/oh-package-lock.json520
-rw-r--r--support/openharmony/oh-package.json513
31 files changed, 625 insertions, 9 deletions
diff --git a/.gitattributes b/.gitattributes
index c08bb1c85e6..5a283db5fe8 100644
--- a/.gitattributes
+++ b/.gitattributes
@@ -1,5 +1,6 @@
* text=auto eol=lf
tests/wpt/** linguist-vendored
+support/openharmony/**/*.ets linguist-language=ts
# Denote all files that are truly binary and should not be modified.
*.png binary
diff --git a/.github/workflows/main.yml b/.github/workflows/main.yml
index 446d4f36ef3..1975a653454 100644
--- a/.github/workflows/main.yml
+++ b/.github/workflows/main.yml
@@ -57,6 +57,7 @@ jobs:
uses: ./.github/workflows/ohos.yml
with:
profile: "release"
+ secrets: inherit
build-result:
name: Result
diff --git a/.github/workflows/ohos.yml b/.github/workflows/ohos.yml
index 61400a33c0f..513efc3c84d 100644
--- a/.github/workflows/ohos.yml
+++ b/.github/workflows/ohos.yml
@@ -55,9 +55,32 @@ jobs:
uses: openharmony-rs/setup-ohos-sdk@v0.1
with:
version: "4.1"
+ components: "native;toolchains;ets;js;previewer"
+ fixup-path: true
+ - name: Install node for hvigor
+ uses: actions/setup-node@v4
+ with:
+ node-version: 18
+ - name: Install hvigor modules
+ run: |
+ mkdir ~/hvigor-installation
+ cd ~/hvigor-installation
+ echo "@ohos:registry=https://repo.harmonyos.com/npm/" > .npmrc
+ npm install "@ohos/hvigor@5" "@ohos/hvigor-ohos-plugin@5"
+ echo "HVIGOR_PATH=$PWD" >> $GITHUB_ENV
+ - name: "Setup HAP signing config"
+ env:
+ SIGNING_MATERIAL: ${{ secrets.SERVO_OHOS_SIGNING_MATERIAL }}
+ if: ${{ env.SIGNING_MATERIAL != '' }} # Allows the build to pass on forks.
+ run: |
+ cd ~
+ echo "${SIGNING_MATERIAL}" | base64 -d > servo-ohos-material.zip
+ unzip servo-ohos-material.zip
+ echo "SERVO_OHOS_SIGNING_CONFIG=${PWD}/servo-ohos-material/signing-configs.json" >> $GITHUB_ENV
- name: Build (arch ${{ matrix.arch }} profile ${{ inputs.profile }})
env:
OHOS_SDK_NATIVE: ${{ steps.setup_sdk.outputs.ohos_sdk_native }}
+ OHOS_BASE_SDK_HOME: ${{ steps.setup_sdk.outputs.ohos-base-sdk-home }}
run: |
python3 ./mach build --locked --target ${{ matrix.arch }} --${{ inputs.profile }}
cp -r target/cargo-timings target/cargo-timings-ohos-${{ matrix.arch }}
@@ -67,3 +90,15 @@ jobs:
name: cargo-timings-ohos-${{ matrix.arch }}
# Using a wildcard here ensures that the archive includes the path.
path: target/cargo-timings-*
+ - name: Upload signed HAP artifact
+ if: ${{ env.SERVO_OHOS_SIGNING_CONFIG != '' }} # Build output has different name if not signed.
+ uses: actions/upload-artifact@v4
+ with:
+ name: ${{ inputs.profile }}-binary-ohos-${{ matrix.arch }}
+ path: target/openharmony/${{ matrix.arch }}/${{ inputs.profile }}/entry/build/default/outputs/default/servoshell-default-signed.hap
+ - name: Upload unsigned HAP artifact
+ if: ${{ env.SERVO_OHOS_SIGNING_CONFIG == '' }} # Build output has different name if not signed.
+ uses: actions/upload-artifact@v4
+ with:
+ name: ${{ inputs.profile }}-binary-ohos-${{ matrix.arch }}
+ path: target/openharmony/${{ matrix.arch }}/${{ inputs.profile }}/entry/build/default/outputs/default/servoshell-default-unsigned.hap
diff --git a/python/servo/command_base.py b/python/servo/command_base.py
index 4359d1c74ca..6513a1b150a 100644
--- a/python/servo/command_base.py
+++ b/python/servo/command_base.py
@@ -297,6 +297,7 @@ class CommandBase(object):
self.config.setdefault("build", {})
self.config["build"].setdefault("android", False)
+ self.config["build"].setdefault("ohos", False)
self.config["build"].setdefault("mode", "")
self.config["build"].setdefault("debug-assertions", False)
self.config["build"].setdefault("debug-mozjs", False)
@@ -328,12 +329,6 @@ class CommandBase(object):
def get_top_dir(self):
return self.context.topdir
- def get_apk_path(self, build_type: BuildType):
- base_path = util.get_target_dir()
- base_path = path.join(base_path, "android", self.target.triple())
- apk_name = "servoapp.apk"
- return path.join(base_path, build_type.directory_name(), apk_name)
-
def get_binary_path(self, build_type: BuildType, asan: bool = False):
base_path = util.get_target_dir()
if asan or self.target.is_cross_build():
@@ -551,7 +546,12 @@ class CommandBase(object):
CommandArgument(
'--android', default=None, action='store_true',
help='Build for Android. If --target is not specified, this '
- 'will choose a default target architecture.',
+ f'will choose the default target architecture ({AndroidTarget.DEFAULT_TRIPLE}).',
+ ),
+ CommandArgument(
+ '--ohos', default=None, action='store_true',
+ help='Build for OpenHarmony. If --target is not specified, this '
+ f'will choose a default target architecture ({OpenHarmonyTarget.DEFAULT_TRIPLE}).',
),
CommandArgument('--win-arm64', action='store_true', help="Use arm64 Windows target"),
CommandArgumentGroup('Feature Selection'),
@@ -646,6 +646,7 @@ class CommandBase(object):
self.configure_build_target(kwargs, suppress_log=True)
kwargs.pop('target', False)
kwargs.pop('android', False)
+ kwargs.pop('ohos', False)
return original_function(self, *args, **kwargs)
return target_configuration_decorator
@@ -686,15 +687,28 @@ class CommandBase(object):
return
android = kwargs.get('android') or self.config["build"]["android"]
+ ohos = kwargs.get('ohos') or self.config["build"]["ohos"]
target_triple = kwargs.get('target')
+ if android and ohos:
+ print("Cannot build both android and ohos targets simultaneously.")
+ sys.exit(1)
+
if android and target_triple:
print("Please specify either --target or --android.")
sys.exit(1)
# Set the default Android target
if android and not target_triple:
- target_triple = "aarch64-linux-android"
+ target_triple = AndroidTarget.DEFAULT_TRIPLE
+
+ if ohos and target_triple:
+ print("Please specify either --target or --ohos.")
+ sys.exit(1)
+
+ # Set the default OpenHarmony target
+ if ohos and not target_triple:
+ target_triple = OpenHarmonyTarget.DEFAULT_TRIPLE
self.target = BuildTarget.from_triple(target_triple)
@@ -705,6 +719,9 @@ class CommandBase(object):
def is_android(self):
return isinstance(self.target, AndroidTarget)
+ def is_openharmony(self):
+ return isinstance(self.target, OpenHarmonyTarget)
+
def is_media_enabled(self, media_stack: Optional[str]):
"""Determine whether media is enabled based on the value of the build target
platform and the value of the '--media-stack' command-line argument.
diff --git a/python/servo/package_commands.py b/python/servo/package_commands.py
index 4b2bef89c01..cad2a73f275 100644
--- a/python/servo/package_commands.py
+++ b/python/servo/package_commands.py
@@ -35,6 +35,7 @@ from servo.command_base import (
archive_deterministically,
BuildNotFound,
cd,
+ check_output,
CommandBase,
is_windows,
)
@@ -124,6 +125,10 @@ class PackageCommands(CommandBase):
default=None,
action='store_true',
help='Package Android')
+ @CommandArgument('--ohos',
+ default=None,
+ action='store_true',
+ help='Package OpenHarmony')
@CommandArgument('--target', '-t',
default=None,
help='Package for given target platform')
@@ -178,6 +183,55 @@ class PackageCommands(CommandBase):
except subprocess.CalledProcessError as e:
print("Packaging Android exited with return value %d" % e.returncode)
return e.returncode
+ elif self.is_openharmony():
+ # hvigor doesn't support an option to place output files in a specific directory
+ # so copy the source files into the target/openharmony directory first.
+ ohos_app_dir = path.join(self.get_top_dir(), "support", "openharmony")
+ build_mode = build_type.directory_name()
+ ohos_target_dir = path.join(
+ self.get_top_dir(), "target", "openharmony", self.target.triple(), build_mode)
+ if path.exists(ohos_target_dir):
+ print("Cleaning up from previous packaging")
+ delete(ohos_target_dir)
+ shutil.copytree(ohos_app_dir, ohos_target_dir)
+
+ # Map non-debug profiles to 'release' buildMode HAP.
+ if build_type.is_custom():
+ build_mode = "release"
+
+ hvigor_command = ["--no-daemon", "assembleHap", "-p", "product=default", "-p", f"buildMode={build_mode}"]
+ # Detect if PATH already has hvigor, or else fallback to npm installation
+ # provided via HVIGOR_PATH
+ if "HVIGOR_PATH" not in env:
+ try:
+ with cd(ohos_target_dir):
+ version = check_output(["hvigorw", "--version", "--no-daemon"])
+ print(f"Found `hvigorw` with version {str(version, 'utf-8').strip()} in system PATH")
+ hvigor_command[0:0] = ["hvigorw"]
+ except FileNotFoundError:
+ print("Unable to find `hvigor` tool. Please either modify PATH to include the"
+ "path to hvigorw or set the HVIGOR_PATH environment variable to the npm"
+ "installation containing `node_modules` directory with hvigor modules.")
+ sys.exit(1)
+ else:
+ env["NODE_PATH"] = env["HVIGOR_PATH"] + "/node_modules"
+ hvigor_script = f"{env['HVIGOR_PATH']}/node_modules/@ohos/hvigor/bin/hvigor.js"
+ hvigor_command[0:0] = ["node", hvigor_script]
+
+ abi_string = self.target.abi_string()
+ ohos_libs_dir = path.join(ohos_target_dir, "entry", "libs", abi_string)
+ os.makedirs(ohos_libs_dir)
+ # The libservoshell.so binary that was built needs to be copied
+ # into the app folder heirarchy where hvigor expects it.
+ print(f"Copying {binary_path} to {ohos_libs_dir}")
+ shutil.copy(binary_path, ohos_libs_dir)
+ try:
+ with cd(ohos_target_dir):
+ print("Calling", hvigor_command)
+ subprocess.check_call(hvigor_command, env=env)
+ except subprocess.CalledProcessError as e:
+ print("Packaging OpenHarmony exited with return value %d" % e.returncode)
+ return e.returncode
elif 'darwin' in self.target.triple():
print("Creating Servo.app")
dir_to_dmg = path.join(target_dir, 'dmg')
@@ -347,6 +401,9 @@ class PackageCommands(CommandBase):
@CommandArgument('--android',
action='store_true',
help='Install on Android')
+ @CommandArgument('--ohos',
+ action='store_true',
+ help='Install on OpenHarmony')
@CommandArgument('--emulator',
action='store_true',
help='For Android, install to the only emulated device')
@@ -376,7 +433,7 @@ class PackageCommands(CommandBase):
return 1
if self.is_android():
- pkg_path = self.get_apk_path(build_type)
+ pkg_path = self.target.get_package_path(build_type.directory_name())
exec_command = [self.android_adb_path(env)]
if emulator and usb:
print("Cannot install to both emulator and USB at the same time.")
@@ -386,6 +443,10 @@ class PackageCommands(CommandBase):
if usb:
exec_command += ["-d"]
exec_command += ["install", "-r", pkg_path]
+ elif self.is_openharmony():
+ pkg_path = self.target.get_package_path(build_type.directory_name())
+ hdc_path = path.join(env["OHOS_SDK_NATIVE"], "../", "toolchains", "hdc")
+ exec_command = [hdc_path, "install", "-r", pkg_path]
elif is_windows():
pkg_path = path.join(path.dirname(binary_path), 'msi', 'Servo.msi')
exec_command = ["msiexec", "/i", pkg_path]
diff --git a/python/servo/platform/build_target.py b/python/servo/platform/build_target.py
index bf795386b5a..ecdb6c71c13 100644
--- a/python/servo/platform/build_target.py
+++ b/python/servo/platform/build_target.py
@@ -21,6 +21,7 @@ from packaging.version import parse as parse_version
from typing import Any, Dict, Optional
import servo.platform
+import servo.util as util
class BuildTarget(object):
@@ -61,6 +62,8 @@ class CrossBuildTarget(BuildTarget):
class AndroidTarget(CrossBuildTarget):
+ DEFAULT_TRIPLE = "aarch64-linux-android"
+
def ndk_configuration(self) -> Dict[str, str]:
target = self.triple()
config = {}
@@ -237,8 +240,16 @@ class AndroidTarget(CrossBuildTarget):
def needs_packaging(self) -> bool:
return True
+ def get_package_path(self, build_type_directory: str) -> str:
+ base_path = util.get_target_dir()
+ base_path = path.join(base_path, "android", self.triple())
+ apk_name = "servoapp.apk"
+ return path.join(base_path, build_type_directory, apk_name)
+
class OpenHarmonyTarget(CrossBuildTarget):
+ DEFAULT_TRIPLE = "aarch64-unknown-linux-ohos"
+
def configure_build_environment(self, env: Dict[str, str], config: Dict[str, Any], topdir: pathlib.Path):
# Paths to OpenHarmony SDK and build tools:
# Note: `OHOS_SDK_NATIVE` is the CMake variable name the `hvigor` build-system
@@ -370,3 +381,17 @@ class OpenHarmonyTarget(CrossBuildTarget):
def needs_packaging(self) -> bool:
return True
+
+ def get_package_path(self, build_type_directory: str) -> str:
+ base_path = util.get_target_dir()
+ base_path = path.join(base_path, "openharmony", self.triple())
+ hap_name = "servoshell-default-signed.hap"
+ build_output_path = path.join("entry", "build", "default", "outputs", "default")
+ return path.join(base_path, build_type_directory, build_output_path, hap_name)
+
+ def abi_string(self) -> str:
+ abi_map = {
+ "aarch64-unknown-linux-ohos": "arm64-v8a",
+ "x86_64-unknown-linux-ohos": "x86_64"
+ }
+ return abi_map[self.triple()]
diff --git a/support/openharmony/AppScope/app.json5 b/support/openharmony/AppScope/app.json5
new file mode 100644
index 00000000000..84df84679d8
--- /dev/null
+++ b/support/openharmony/AppScope/app.json5
@@ -0,0 +1,10 @@
+{
+ "app": {
+ "bundleName": "org.servo.servoshell",
+ "vendor": "example",
+ "versionCode": 1000000,
+ "versionName": "1.0.0",
+ "icon": "$media:servo_64",
+ "label": "$string:app_name"
+ }
+}
diff --git a/support/openharmony/AppScope/resources/base/element/string.json b/support/openharmony/AppScope/resources/base/element/string.json
new file mode 100644
index 00000000000..98edfcdb0fb
--- /dev/null
+++ b/support/openharmony/AppScope/resources/base/element/string.json
@@ -0,0 +1,8 @@
+{
+ "string": [
+ {
+ "name": "app_name",
+ "value": "Servo Shell"
+ }
+ ]
+}
diff --git a/support/openharmony/AppScope/resources/base/media/servo_64.png b/support/openharmony/AppScope/resources/base/media/servo_64.png
new file mode 120000
index 00000000000..0361142fb10
--- /dev/null
+++ b/support/openharmony/AppScope/resources/base/media/servo_64.png
@@ -0,0 +1 @@
+../../../../../../resources/servo_64.png \ No newline at end of file
diff --git a/support/openharmony/build-profile.json5 b/support/openharmony/build-profile.json5
new file mode 100644
index 00000000000..0d5ab09ba42
--- /dev/null
+++ b/support/openharmony/build-profile.json5
@@ -0,0 +1,44 @@
+{
+ "app": {
+ "products": [
+ {
+ "name": "default",
+ "signingConfig": "default",
+ "compileSdkVersion": 11,
+ "compatibleSdkVersion": 11,
+ "targetSdkVersion": 11,
+ "runtimeOS": "OpenHarmony"
+ },
+ {
+ "name": "harmonyos",
+ "signingConfig": "default",
+ "compatibleSdkVersion": "4.1.0(11)",
+ "targetSdkVersion": "4.1.0(11)",
+ "runtimeOS": "HarmonyOS"
+ }
+ ],
+ "buildModeSet": [
+ {
+ "name": "debug"
+ },
+ {
+ "name": "release"
+ }
+ ]
+ },
+ "modules": [
+ {
+ "name": "servoshell",
+ "srcPath": "./entry",
+ "targets": [
+ {
+ "name": "default",
+ "applyToProducts": [
+ "default",
+ "harmonyos"
+ ]
+ }
+ ]
+ }
+ ]
+}
diff --git a/support/openharmony/entry/build-profile.json5 b/support/openharmony/entry/build-profile.json5
new file mode 100644
index 00000000000..837a01379a2
--- /dev/null
+++ b/support/openharmony/entry/build-profile.json5
@@ -0,0 +1,36 @@
+{
+ "apiType": "stageMode",
+ "buildOption": {
+ "arkOptions": {
+ },
+ "externalNativeOptions": {
+ "path": "./src/main/cpp/CMakeLists.txt",
+ "arguments": "",
+ "cppFlags": "",
+ "abiFilters": ["arm64-v8a", "x86_64"]
+ }
+ },
+ "buildOptionSet": [
+ {
+ "name": "release",
+ "arkOptions": {
+ "obfuscation": {
+ "ruleOptions": {
+ "enable": true,
+ "files": [
+ "./obfuscation-rules.txt"
+ ]
+ }
+ }
+ }
+ },
+ ],
+ "targets": [
+ {
+ "name": "default"
+ },
+ {
+ "name": "ohosTest",
+ }
+ ]
+}
diff --git a/support/openharmony/entry/hvigorfile.ts b/support/openharmony/entry/hvigorfile.ts
new file mode 100644
index 00000000000..c6edcd90486
--- /dev/null
+++ b/support/openharmony/entry/hvigorfile.ts
@@ -0,0 +1,6 @@
+import { hapTasks } from '@ohos/hvigor-ohos-plugin';
+
+export default {
+ system: hapTasks, /* Built-in plugin of Hvigor. It cannot be modified. */
+ plugins:[] /* Custom plugin to extend the functionality of Hvigor. */
+}
diff --git a/support/openharmony/entry/obfuscation-rules.txt b/support/openharmony/entry/obfuscation-rules.txt
new file mode 100644
index 00000000000..985b2aeb765
--- /dev/null
+++ b/support/openharmony/entry/obfuscation-rules.txt
@@ -0,0 +1,18 @@
+# Define project specific obfuscation rules here.
+# You can include the obfuscation configuration files in the current module's build-profile.json5.
+#
+# For more details, see
+# https://gitee.com/openharmony/arkcompiler_ets_frontend/blob/master/arkguard/README.md
+
+# Obfuscation options:
+# -disable-obfuscation: disable all obfuscations
+# -enable-property-obfuscation: obfuscate the property names
+# -enable-toplevel-obfuscation: obfuscate the names in the global scope
+# -compact: remove unnecessary blank spaces and all line feeds
+# -remove-log: remove all console.* statements
+# -print-namecache: print the name cache that contains the mapping from the old names to new names
+# -apply-namecache: reuse the given cache file
+
+# Keep options:
+# -keep-property-name: specifies property names that you want to keep
+# -keep-global-name: specifies names that you want to keep in the global scope \ No newline at end of file
diff --git a/support/openharmony/entry/oh-package.json5 b/support/openharmony/entry/oh-package.json5
new file mode 100644
index 00000000000..0458a607574
--- /dev/null
+++ b/support/openharmony/entry/oh-package.json5
@@ -0,0 +1,12 @@
+{
+ "license": "Apache-2.0",
+ "devDependencies": {},
+ "author": "",
+ "name": "servoshell",
+ "description": "Please describe the basic information.",
+ "main": "",
+ "version": "1.0.0",
+ "dependencies": {
+ "libservoshell.so": "file:./src/main/cpp/types/libentry"
+ }
+}
diff --git a/support/openharmony/entry/src/main/cpp/CMakeLists.txt b/support/openharmony/entry/src/main/cpp/CMakeLists.txt
new file mode 100644
index 00000000000..3c68b0d074b
--- /dev/null
+++ b/support/openharmony/entry/src/main/cpp/CMakeLists.txt
@@ -0,0 +1,3 @@
+# the minimum version of CMake.
+cmake_minimum_required(VERSION 3.4.1)
+project(servoshell)
diff --git a/support/openharmony/entry/src/main/cpp/types/libentry/index.d.ts b/support/openharmony/entry/src/main/cpp/types/libentry/index.d.ts
new file mode 100644
index 00000000000..a866293df8e
--- /dev/null
+++ b/support/openharmony/entry/src/main/cpp/types/libentry/index.d.ts
@@ -0,0 +1 @@
+export const loadURL: (url: string) => void; \ No newline at end of file
diff --git a/support/openharmony/entry/src/main/cpp/types/libentry/oh-package.json5 b/support/openharmony/entry/src/main/cpp/types/libentry/oh-package.json5
new file mode 100644
index 00000000000..a6f6a3f9921
--- /dev/null
+++ b/support/openharmony/entry/src/main/cpp/types/libentry/oh-package.json5
@@ -0,0 +1,6 @@
+{
+ "name": "libservoshell.so",
+ "types": "./index.d.ts",
+ "version": "",
+ "description": "The native module which exposes a C API to servoshell"
+}
diff --git a/support/openharmony/entry/src/main/ets/entryability/EntryAbility.ets b/support/openharmony/entry/src/main/ets/entryability/EntryAbility.ets
new file mode 100644
index 00000000000..bd1f2320984
--- /dev/null
+++ b/support/openharmony/entry/src/main/ets/entryability/EntryAbility.ets
@@ -0,0 +1,43 @@
+import AbilityConstant from '@ohos.app.ability.AbilityConstant';
+import hilog from '@ohos.hilog';
+import UIAbility from '@ohos.app.ability.UIAbility';
+import Want from '@ohos.app.ability.Want';
+import window from '@ohos.window';
+
+export default class EntryAbility extends UIAbility {
+ onCreate(want: Want, launchParam: AbilityConstant.LaunchParam) {
+ hilog.info(0x0000, 'testTag', '%{public}s', 'Ability onCreate');
+ }
+
+ onDestroy() {
+ hilog.info(0x0000, 'testTag', '%{public}s', 'Ability onDestroy');
+ }
+
+ onWindowStageCreate(windowStage: window.WindowStage) {
+ // Main window is created, set main page for this ability
+ hilog.info(0x0000, 'testTag', '%{public}s', 'Ability onWindowStageCreate');
+
+ windowStage.loadContent('pages/Index', (err, data) => {
+ if (err.code) {
+ hilog.error(0x0000, 'testTag', 'Failed to load the content. Cause: %{public}s', JSON.stringify(err) ?? '');
+ return;
+ }
+ hilog.info(0x0000, 'testTag', 'Succeeded in loading the content. Data: %{public}s', JSON.stringify(data) ?? '');
+ });
+ }
+
+ onWindowStageDestroy() {
+ // Main window is destroyed, release UI related resources
+ hilog.info(0x0000, 'testTag', '%{public}s', 'Ability onWindowStageDestroy');
+ }
+
+ onForeground() {
+ // Ability has brought to foreground
+ hilog.info(0x0000, 'testTag', '%{public}s', 'Ability onForeground');
+ }
+
+ onBackground() {
+ // Ability has back to background
+ hilog.info(0x0000, 'testTag', '%{public}s', 'Ability onBackground');
+ }
+};
diff --git a/support/openharmony/entry/src/main/ets/pages/Index.ets b/support/openharmony/entry/src/main/ets/pages/Index.ets
new file mode 100644
index 00000000000..b3b711683ea
--- /dev/null
+++ b/support/openharmony/entry/src/main/ets/pages/Index.ets
@@ -0,0 +1,85 @@
+import display from '@ohos.display';
+import deviceInfo from '@ohos.deviceInfo';
+
+interface ServoXComponentInterface {
+ loadURL(url: string): void;
+ registerURLcallback(callback: (url: string) => void): void;
+ initServo(options: InitOpts): void;
+}
+
+interface InitOpts {
+ url: string;
+ deviceType: string,
+ osFullName: string,
+ displayDensity: number,
+}
+
+function get_density(): number {
+ try {
+ let displayClass = display.getDefaultDisplaySync();
+ console.info('Test densityDPI:' + JSON.stringify(displayClass.densityDPI));
+ return displayClass.densityDPI / 160;
+ } catch (exception) {
+ console.error('Failed to obtain the default display object. Code: ' + JSON.stringify(exception));
+ return 3;
+ }
+}
+
+function get_device_type(): string {
+ let device_type: string = deviceInfo.deviceType;
+ if (device_type == "") {
+ console.error("deviceInfo.deviceType is empty string!")
+ } else {
+ console.info("Device type is " + device_type)
+ }
+ return device_type;
+}
+
+@Entry
+@Component
+struct Index {
+ xComponentContext: ServoXComponentInterface | undefined = undefined;
+ xComponentAttrs: XComponentAttrs = {
+ id: 'ServoDemo',
+ type: XComponentType.SURFACE,
+ libraryname: 'servoshell',
+ }
+ @State urlToLoad: string = 'https://servo.org'
+
+ build() {
+ Column() {
+ TextInput({placeholder:'URL',text: $$this.urlToLoad})
+ .type(InputType.Normal)
+ .onChange((value) => {
+ this.urlToLoad = value
+ })
+ .onSubmit((EnterKeyType)=>{
+ this.xComponentContext?.loadURL(this.urlToLoad)
+ console.info('Load URL: ', this.urlToLoad)
+ })
+ XComponent(this.xComponentAttrs)
+ .focusable(true)
+ .onLoad((xComponentContext) => {
+ this.xComponentContext = xComponentContext as ServoXComponentInterface;
+ let init_options: InitOpts = {
+ url: this.urlToLoad,
+ deviceType: get_device_type(),
+ osFullName: deviceInfo.osFullName,
+ displayDensity: get_density()
+ }
+ this.xComponentContext.initServo(init_options)
+ this.xComponentContext.registerURLcallback((new_url) => {
+ console.info('New URL from native: ', new_url)
+ this.urlToLoad = new_url
+ })
+ })
+ }
+ .width('100%')
+ }
+}
+
+interface XComponentAttrs {
+ id: string;
+ type: number;
+ libraryname: string;
+} \ No newline at end of file
diff --git a/support/openharmony/entry/src/main/module.json5 b/support/openharmony/entry/src/main/module.json5
new file mode 100644
index 00000000000..806334680ec
--- /dev/null
+++ b/support/openharmony/entry/src/main/module.json5
@@ -0,0 +1,52 @@
+{
+ "module": {
+ "name": "servoshell",
+ "type": "entry",
+ "description": "$string:module_desc",
+ "mainElement": "EntryAbility",
+ "deviceTypes": [
+ "default"
+ ],
+ "deliveryWithInstall": true,
+ "installationFree": false,
+ "pages": "$profile:main_pages",
+ "abilities": [
+ {
+ "name": "EntryAbility",
+ "srcEntry": "./ets/entryability/EntryAbility.ets",
+ "description": "$string:EntryAbility_desc",
+ "icon": "$media:servo_1024",
+ "label": "$string:EntryAbility_label",
+ "startWindowIcon": "$media:servo_64",
+ "startWindowBackground": "$color:start_window_background",
+ "exported": true,
+ "skills": [
+ {
+ "entities": [
+ "entity.system.home"
+ ],
+ "actions": [
+ "action.system.home"
+ ]
+ }
+ ]
+ }
+ ],
+ // ohos.permission.RUN_DYN_CODE - required for JIT?
+ "requestPermissions": [
+ {
+ "name": "ohos.permission.GET_NETWORK_INFO",
+ "reason": "$string:perm_reason",
+ },
+ {
+ "name": "ohos.permission.INTERNET",
+ "reason": "$string:perm_reason",
+ },
+ {
+ "name": "ohos.permission.KEEP_BACKGROUND_RUNNING",
+ "reason": "$string:perm_reason",
+ },
+
+ ]
+ }
+}
diff --git a/support/openharmony/entry/src/main/resources/base/element/color.json b/support/openharmony/entry/src/main/resources/base/element/color.json
new file mode 100644
index 00000000000..162a7b6f4af
--- /dev/null
+++ b/support/openharmony/entry/src/main/resources/base/element/color.json
@@ -0,0 +1,8 @@
+{
+ "color": [
+ {
+ "name": "start_window_background",
+ "value": "#FFFFFF"
+ }
+ ]
+}
diff --git a/support/openharmony/entry/src/main/resources/base/element/string.json b/support/openharmony/entry/src/main/resources/base/element/string.json
new file mode 100644
index 00000000000..567cdd8ba16
--- /dev/null
+++ b/support/openharmony/entry/src/main/resources/base/element/string.json
@@ -0,0 +1,20 @@
+{
+ "string": [
+ {
+ "name": "module_desc",
+ "value": "module description"
+ },
+ {
+ "name": "EntryAbility_desc",
+ "value": "description"
+ },
+ {
+ "name": "EntryAbility_label",
+ "value": "label"
+ },
+ {
+ "name": "perm_reason",
+ "value": "This is a demo app, and we need permissions"
+ }
+ ]
+}
diff --git a/support/openharmony/entry/src/main/resources/base/media/servo_1024.png b/support/openharmony/entry/src/main/resources/base/media/servo_1024.png
new file mode 120000
index 00000000000..c5ecb1ae891
--- /dev/null
+++ b/support/openharmony/entry/src/main/resources/base/media/servo_1024.png
@@ -0,0 +1 @@
+../../../../../../../../resources/servo_1024.png \ No newline at end of file
diff --git a/support/openharmony/entry/src/main/resources/base/media/servo_64.png b/support/openharmony/entry/src/main/resources/base/media/servo_64.png
new file mode 120000
index 00000000000..0cc4be4d7ab
--- /dev/null
+++ b/support/openharmony/entry/src/main/resources/base/media/servo_64.png
@@ -0,0 +1 @@
+../../../../../../../../resources/servo_64.png \ No newline at end of file
diff --git a/support/openharmony/entry/src/main/resources/base/profile/main_pages.json b/support/openharmony/entry/src/main/resources/base/profile/main_pages.json
new file mode 100644
index 00000000000..1898d94f58d
--- /dev/null
+++ b/support/openharmony/entry/src/main/resources/base/profile/main_pages.json
@@ -0,0 +1,5 @@
+{
+ "src": [
+ "pages/Index"
+ ]
+}
diff --git a/support/openharmony/entry/src/main/resources/en_US/element/string.json b/support/openharmony/entry/src/main/resources/en_US/element/string.json
new file mode 100644
index 00000000000..efbe436fe0d
--- /dev/null
+++ b/support/openharmony/entry/src/main/resources/en_US/element/string.json
@@ -0,0 +1,16 @@
+{
+ "string": [
+ {
+ "name": "module_desc",
+ "value": "module description"
+ },
+ {
+ "name": "EntryAbility_desc",
+ "value": "description"
+ },
+ {
+ "name": "EntryAbility_label",
+ "value": "Servo"
+ }
+ ]
+}
diff --git a/support/openharmony/entry/src/main/resources/zh_CN/element/string.json b/support/openharmony/entry/src/main/resources/zh_CN/element/string.json
new file mode 100644
index 00000000000..96a3f5b0dfa
--- /dev/null
+++ b/support/openharmony/entry/src/main/resources/zh_CN/element/string.json
@@ -0,0 +1,16 @@
+{
+ "string": [
+ {
+ "name": "module_desc",
+ "value": "模块描述"
+ },
+ {
+ "name": "EntryAbility_desc",
+ "value": "description"
+ },
+ {
+ "name": "EntryAbility_label",
+ "value": "Servo"
+ }
+ ]
+}
diff --git a/support/openharmony/hvigor/hvigor-config.json5 b/support/openharmony/hvigor/hvigor-config.json5
new file mode 100644
index 00000000000..9bb1d298f85
--- /dev/null
+++ b/support/openharmony/hvigor/hvigor-config.json5
@@ -0,0 +1,17 @@
+{
+ "modelVersion": "5.0.0",
+ "dependencies": {
+ },
+ "execution": {
+ // "daemon": true, /* Enable daemon compilation. Default: true */
+ // "incremental": true, /* Enable incremental compilation. Default: true */
+ // "parallel": true, /* Enable parallel compilation. Default: true */
+ // "typeCheck": false, /* Enable typeCheck. Default: false */
+ },
+ "logging": {
+ // "level": "info" /* Define the log level. Value: [ "debug" | "info" | "warn" | "error" ]. Default: "info" */
+ },
+ "debugging": {
+ // "stacktrace": false /* Disable stacktrace compilation. Default: false */
+ }
+}
diff --git a/support/openharmony/hvigorfile.ts b/support/openharmony/hvigorfile.ts
new file mode 100644
index 00000000000..bcdfc0fcef9
--- /dev/null
+++ b/support/openharmony/hvigorfile.ts
@@ -0,0 +1,34 @@
+import { appTasks, OhosAppContext, OhosPluginId } from '@ohos/hvigor-ohos-plugin';
+import { getNode } from '@ohos/hvigor'
+import * as fs from 'fs';
+import * as path from 'path';
+
+const rootNode = getNode(__filename);
+rootNode.afterNodeEvaluate(node => {
+ const appContext = node.getContext(OhosPluginId.OHOS_APP_PLUGIN) as OhosAppContext;
+ const buildProfileOpt = appContext.getBuildProfileOpt();
+ const signingConfigsPath = process.env["SERVO_OHOS_SIGNING_CONFIG"];
+ if (signingConfigsPath) {
+ if (!fs.existsSync(signingConfigsPath)) {
+ console.error("File referenced by SERVO_OHOS_SIGNING_CONFIG does not exist!");
+ return;
+ }
+ const basePath = path.dirname(signingConfigsPath);
+ const signingConfigs = JSON.parse(fs.readFileSync(signingConfigsPath));
+ for (const config of signingConfigs) {
+ config.material.certpath = path.resolve(basePath, config.material.certpath);
+ config.material.profile = path.resolve(basePath, config.material.profile);
+ config.material.storeFile = path.resolve(basePath, config.material.storeFile);
+ }
+ buildProfileOpt['app']['signingConfigs'] = signingConfigs;
+ } else {
+ console.log('Signing config not found in enviroment. hvigor will fallback to build-profile.json5.')
+ }
+ // Set the obj object back to the context object to enable the build process and results.
+ appContext.setBuildProfileOpt(buildProfileOpt);
+})
+
+export default {
+ system: appTasks, /* Built-in plugin of Hvigor. It cannot be modified. */
+ plugins:[] /* Custom plugin to extend the functionality of Hvigor. */
+}
diff --git a/support/openharmony/oh-package-lock.json5 b/support/openharmony/oh-package-lock.json5
new file mode 100644
index 00000000000..2613377bbe0
--- /dev/null
+++ b/support/openharmony/oh-package-lock.json5
@@ -0,0 +1,20 @@
+{
+ "meta": {
+ "stableOrder": true
+ },
+ "lockfileVersion": 3,
+ "ATTENTION": "THIS IS AN AUTOGENERATED FILE. DO NOT EDIT THIS FILE DIRECTLY.",
+ "specifiers": {
+ "@ohos/hypium@1.0.6": "@ohos/hypium@1.0.6"
+ },
+ "packages": {
+ "@ohos/hypium@1.0.6": {
+ "name": "@ohos/hypium",
+ "version": "1.0.6",
+ "integrity": "sha512-IgF26ui+ZL2/5ITpAoHqtNbpdDOyoXTjK8TEyncETNyH3gzklKD9+ZPdsxEvNVRbigiptmisfugNBNFM3OdpeQ==",
+ "resolved": "https://cmc.centralrepo.rnd.huawei.com/artifactory/api/npm/product_npm/@ohos/hypium/-/@ohos/hypium-1.0.6.tgz",
+ "shasum": "c2042745b131325ccf3d2b7eb7f77306758a838b",
+ "registryType": "npm"
+ }
+ }
+} \ No newline at end of file
diff --git a/support/openharmony/oh-package.json5 b/support/openharmony/oh-package.json5
new file mode 100644
index 00000000000..0875ceb5b8d
--- /dev/null
+++ b/support/openharmony/oh-package.json5
@@ -0,0 +1,13 @@
+{
+ "modelVersion": "5.0.0",
+ "license": "Apache-2.0",
+ "devDependencies": {
+ "@ohos/hypium": "1.0.6"
+ },
+ "author": "",
+ "name": "servoshell",
+ "description": "Servo browser shell",
+ "main": "",
+ "version": "1.0.0",
+ "dependencies": {}
+}