From fa4d172fad896ee102fc419b14055733ecc73c0c Mon Sep 17 00:00:00 2001 From: Xiaoming Shi Date: Fri, 13 Dec 2024 17:36:33 -0800 Subject: [PATCH] (WIP, DNS) [android] Support snapshot of media implementation b/327287075 --- starboard/android/shared/BUILD.gn | 4 + starboard/android/shared/drm_system.cc | 2 +- .../shared/media_snapshot/media_snapshot.h | 27 ++ starboard/tools/media/find_dependency.py | 65 +++ starboard/tools/media/gn_utils.py | 179 ++++++++ starboard/tools/media/snapshot.py | 400 ++++++++++++++++++ starboard/tools/media/source_utils.py | 287 +++++++++++++ starboard/tools/media/utils.py | 112 +++++ 8 files changed, 1075 insertions(+), 1 deletion(-) create mode 100644 starboard/shared/media_snapshot/media_snapshot.h create mode 100644 starboard/tools/media/find_dependency.py create mode 100644 starboard/tools/media/gn_utils.py create mode 100644 starboard/tools/media/snapshot.py create mode 100644 starboard/tools/media/source_utils.py create mode 100644 starboard/tools/media/utils.py diff --git a/starboard/android/shared/BUILD.gn b/starboard/android/shared/BUILD.gn index cf4b70c4dd9f..a83ceac47638 100644 --- a/starboard/android/shared/BUILD.gn +++ b/starboard/android/shared/BUILD.gn @@ -352,6 +352,10 @@ static_library("starboard_platform") { if (sb_evergreen_compatible_use_libunwind) { deps += [ "//third_party/llvm-project/libunwind:unwind_starboard" ] } + + snapshotted_media_files = [] + + sources -= snapshotted_media_files } static_library("starboard_base_symbolize") { diff --git a/starboard/android/shared/drm_system.cc b/starboard/android/shared/drm_system.cc index 156155566bf7..d070f774c85a 100644 --- a/starboard/android/shared/drm_system.cc +++ b/starboard/android/shared/drm_system.cc @@ -67,7 +67,7 @@ SbDrmSessionRequestType SbDrmSessionRequestTypeFromMediaDrmKeyRequestType( // This has to be defined outside the above anonymous namespace to be picked up // by the comparison of std::vector. -bool operator==(const SbDrmKeyId& left, const SbDrmKeyId& right) { +inline bool operator==(const SbDrmKeyId& left, const SbDrmKeyId& right) { if (left.identifier_size != right.identifier_size) { return false; } diff --git a/starboard/shared/media_snapshot/media_snapshot.h b/starboard/shared/media_snapshot/media_snapshot.h new file mode 100644 index 000000000000..bba384092949 --- /dev/null +++ b/starboard/shared/media_snapshot/media_snapshot.h @@ -0,0 +1,27 @@ +// Copyright 2024 The Cobalt Authors. All Rights Reserved. +// +// 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. + +#ifndef STARBOARD_SHARED_MEDIA_SNAPSHOT_MEDIA_SNAPSHOT_H_ +#define STARBOARD_SHARED_MEDIA_SNAPSHOT_MEDIA_SNAPSHOT_H_ + +inline int GetMediaSnapshotVersion() { +#if SB_API_VERSION >= 15 + return 2500; +#else // SB_API_VERSION >= 15 + // Media snapshot is only support for C25 or after. + return 0; +#endif // SB_API_VERSION >= 15 +} + +#endif // STARBOARD_SHARED_MEDIA_SNAPSHOT_MEDIA_SNAPSHOT_H_ diff --git a/starboard/tools/media/find_dependency.py b/starboard/tools/media/find_dependency.py new file mode 100644 index 000000000000..5eec34c28106 --- /dev/null +++ b/starboard/tools/media/find_dependency.py @@ -0,0 +1,65 @@ +#!/usr/bin/env python3 +# Copyright 2024 The Cobalt Authors. All Rights Reserved. +# +# 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. +'''Create a snapshot of an Starboard Android TV implementation under + 'starboard/android/shared/media_/'. This helps with running + multiple Starboard media implementations side by side.''' + +import gn_utils +import os +import source_utils +import utils + +_GN_TARGETS = [ + '//starboard/common:common', + '//starboard/android/shared:starboard_platform', + '//starboard/shared/starboard/media:media_util', + '//starboard/shared/starboard/player/filter:filter_based_player_sources', +] + + +def find_inbound_dependencies(project_root_dir, ninja_output_pathname): + project_root_dir = os.path.abspath(os.path.expanduser(project_root_dir)) + assert os.path.isdir(project_root_dir) + assert os.path.isdir(os.path.join(project_root_dir, ninja_output_pathname)) + + source_files = [] + + for target in _GN_TARGETS: + source_files += gn_utils.get_source_pathnames(project_root_dir, + ninja_output_pathname, target) + + source_files.sort() + + non_media_files = [f for f in source_files if not utils.is_media_file(f)] + + inbound_dependencies = {} + + for file in non_media_files: + with open(file, encoding='utf-8') as f: + content = f.read() + + headers = source_utils.extract_project_includes(content) + for header in headers: + if utils.is_media_file(header): + if header in inbound_dependencies: + inbound_dependencies[header].append(file) + else: + inbound_dependencies[header] = [file] + + for header, sources in inbound_dependencies.items(): + print(header) + for source in sources: + print(' ', source) + print() diff --git a/starboard/tools/media/gn_utils.py b/starboard/tools/media/gn_utils.py new file mode 100644 index 000000000000..aab52501ec1b --- /dev/null +++ b/starboard/tools/media/gn_utils.py @@ -0,0 +1,179 @@ +#!/usr/bin/env python3 +# Copyright 2024 The Cobalt Authors. All Rights Reserved. +# +# 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. +'''Utility to work with gn.''' + +from datetime import datetime +from textwrap import dedent + +import os +import subprocess + +_COPYRIGHT_HEADER = '''\ + # Copyright {0} The Cobalt Authors. All Rights Reserved. + # + # 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. + ''' + +_GN_CONTENT = '''\ +static_library("{0}") {{ + check_includes = false + + sources = [ + {1} + ] + + configs += [ "//starboard/build/config:starboard_implementation" ] + + public_deps = [ + "//starboard/common", + ] + deps = [ + "//base", # TODO: Remove once the upstream is refined. + "//third_party/opus", + ] +}} +''' + +_CANONICAL_GN_CONTENT = ''' +static_library("media_snapshot") {{ + check_includes = false + + sources = [ + {0} + ] + + configs += [ "//starboard/build/config:starboard_implementation" ] + + public_deps = [ + "//starboard/common", + ] +}} +''' + + +def _get_copyright_header(): + return dedent(_COPYRIGHT_HEADER).format(datetime.now().year) + + +def convert_source_list_to_gn_format(project_root_dir, gn_pathname, + file_pathnames): + abs_project_root_dir = os.path.abspath(project_root_dir) + abs_gn_pathname = os.path.abspath(gn_pathname) + + # The gn file should reside in the project dir. + assert abs_gn_pathname.find(abs_project_root_dir) == 0 + + source_list = [] + + for file_pathname in file_pathnames: + abs_file_pathname = os.path.abspath(file_pathname) + if os.path.dirname(file_pathname) == os.path.dirname(abs_gn_pathname): + rel_file_pathname = os.path.basename(file_pathname) + else: + rel_file_pathname = '//' + os.path.relpath(abs_file_pathname, + abs_project_root_dir) + source_list.append('\"' + rel_file_pathname + '",') + + source_list.sort() + + return source_list + + +def create_gn_file(project_root_dir, gn_pathname, library_name, file_pathnames): + abs_project_root_dir = os.path.abspath(project_root_dir) + abs_gn_pathname = os.path.abspath(gn_pathname) + + # The gn file should reside in the project dir. + assert abs_gn_pathname.find(abs_project_root_dir) == 0 + + source_list = convert_source_list_to_gn_format(project_root_dir, gn_pathname, + file_pathnames) + + with open(abs_gn_pathname, 'w+', encoding='utf-8') as f: + f.write(_get_copyright_header() + '\n' + + _GN_CONTENT.format(library_name, '\n '.join(source_list))) + + +def _get_full_pathname(project_root_dir, pathname_in_gn_format): + ''' Transform a pathname in gn format to unix format + + project_root_dir: The project root directory in unix format, e.g. + '/home/.../cobalt' + pathname_in_gn_format: A pathname in gn format, e.g. '//starboard/media.h' + return: the full path name as '/home/.../cobalt/starboard/media.h' + ''' + assert pathname_in_gn_format.find('//') == 0 + pathname_in_gn_format = pathname_in_gn_format[2:] + pathname = os.path.join(project_root_dir, pathname_in_gn_format) + if pathname.find('game-activity') < 0: + assert os.path.isfile(pathname), pathname + return pathname + + +def get_source_pathnames(project_root_dir, ninja_root_dir, + target_name_or_names): + ''' Return a list of source files built for a particular ninja target or a + series of target names. + + project_root_dir: The project root directory, e.g. '/home/.../cobalt' + ninja_root_dir: The output directory, e.g. 'out/android-arm' + target_name_or_names: The name of the ninja target, e.g. + '//cobalt/base:base', or a series pf target names, e.g. + ('//cobalt/base:base', '//cobalt/media:media'). + ''' + if isinstance(target_name_or_names, str): + target_name = target_name_or_names + else: + # Assume `target_name_or_names` is a container + source_files = [] + for target_name in target_name_or_names: + source_files += get_source_pathnames(project_root_dir, ninja_root_dir, + target_name) + return source_files + + saved_python_path = os.environ['PYTHONPATH'] + os.environ['PYTHONPATH'] = os.path.abspath( + project_root_dir) + ':' + os.environ['PYTHONPATH'] + gn_desc = subprocess.check_output(['gn', 'desc', ninja_root_dir, target_name], + cwd=project_root_dir).decode('utf-8') + os.environ['PYTHONPATH'] = saved_python_path + + # gn_desc is in format: + # ... + # sources + # //path/name1 + # //path/name2 + # + # ... + lines = gn_desc.split('\n') + sources_index = lines.index('sources') + assert sources_index >= 0 + sources_index += 1 + sources = [] + while sources_index < len(lines) and lines[sources_index]: + sources.append( + _get_full_pathname(project_root_dir, lines[sources_index].strip())) + sources_index += 1 + return sources diff --git a/starboard/tools/media/snapshot.py b/starboard/tools/media/snapshot.py new file mode 100644 index 000000000000..f4a3ddd081fd --- /dev/null +++ b/starboard/tools/media/snapshot.py @@ -0,0 +1,400 @@ +#!/usr/bin/env python3 +# Copyright 2024 The Cobalt Authors. All Rights Reserved. +# +# 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. +'''Create a snapshot of an Starboard Android TV implementation under + 'starboard/android/shared/media_snapshot_/'. This helps with + running multiple Starboard media implementations side by side.''' + +import gn_utils +import os +import source_utils +import utils + + +class Snapshot: + ''' Snapshot an SbPlayer implementation on Android TV + + It creates a snapshot of an SbPlayer implementation on Android TV + specified by 'source_project_root_dir' and `_BUILD_CONFIG` into + '/starboard/android/shared/ + '. + + The snapshot can be used side by side with the default SbPlayer + implementation on Android TV of ''.''' + + _BUILD_CONFIG = 'out/android-arm_devel' + + _GN_TARGETS = [ + '//starboard/common:common', + '//starboard/android/shared:starboard_platform', + '//starboard/shared/starboard/media:media_util', + '//starboard/shared/starboard/player/filter:filter_based_player_sources', + ] + + _DESTINATION_SUB_DIR = 'starboard/android/shared' + + def __init__(self, source_project_root_dir, destination_project_root_dir, + media_snapshot_version, verbose): + ''' Constructs an object used to create a snapshot of an SbPlayer + implementation on Android TV specified by 'source_project_root_dir' + and `_BUILD_CONFIG` into '/ + starboard/android/shared/'. + + For example, when called with '~/chromium_clean/src', '~/chromium/src', + 'out/android-arm_devel', and '2500', the object will create an Android + TV SbPlayer implementation under the destination project inside + '~/chromium/src/starboard/android/shared/media_snapshot_2500' according + to the files used in the source project using android-arm devel build. + It will also adjust gn references to the newly created project. + + Note that gn.py has to be called on both projects before calling this + function. ''' + + # In case an integer version like 2500 is passed in accidentally. + assert isinstance(media_snapshot_version, str) + + self.source_project_root_dir = os.path.abspath( + os.path.expanduser(source_project_root_dir)) + self.destination_project_root_dir = os.path.abspath( + os.path.expanduser(destination_project_root_dir)) + self.media_snapshot_version = media_snapshot_version + self.verbose = verbose + + assert os.path.isdir(self.source_project_root_dir) + assert os.path.isdir( + os.path.join(self.source_project_root_dir, self._BUILD_CONFIG)) + assert os.path.isdir(self.destination_project_root_dir) + # The snapshot process may modify some of the destination files. Requiring + # a separate checkout as source folder to avoid accidentally snapshotting + # modified source files. + assert not os.path.samefile(self.source_project_root_dir, + self.destination_project_root_dir) + # Some of the logic relies on snapshotted files are inside a subfolder + # containing 'media_snapshot/'. This will break if in the unlikely case + # that the root dir of the destination project contains the string + # 'media_snapshot/'. + assert self.destination_project_root_dir.find('media_snapshot/') == -1 + + def _get_destination_pathname(self, source_pathname): + ''' If the source is 'starboard/shared/starboard/media/media_util.cc', the + destination will be + 'starboard/android/shared/media_snapshot//shared/starboard/ + media/media_util/media_util.cc'. + Note that the leading 'starboard/' of the source will be removed. ''' + + source_rel_pathname = os.path.relpath(source_pathname, + self.source_project_root_dir) + # Only works with Starboard implementation files + assert source_rel_pathname.find('starboard' + os.path.sep) == 0 + + source_rel_pathname = source_rel_pathname[len('starboard') + 1:] + + return os.path.join(self.destination_project_root_dir, + self._DESTINATION_SUB_DIR, + 'media_snapshot/' + self.media_snapshot_version, + source_rel_pathname) + + def snapshot(self): + self._try_create_canonical_sb_implementation() + + print('Gathering source files ...') + source_files = gn_utils.get_source_pathnames(self.source_project_root_dir, + self._BUILD_CONFIG, + self._GN_TARGETS) + source_files = [f for f in source_files if utils.is_media_file(f)] + source_files.sort() + + destination_files = [] + + class_names = [] + headers_dict = {} + + for source_file in source_files: + destination_file = self._get_destination_pathname(source_file) + if source_utils.is_header_file(source_file): + headers_dict[os.path.relpath( + source_file, self.source_project_root_dir)] = os.path.relpath( + destination_file, self.destination_project_root_dir) + with open(source_file, encoding='utf-8') as f: + content = f.read() + class_names += source_utils.extract_class_or_struct_names(content) + + destination_files.append(destination_file) + + class_names.sort() + + print('Snapshotting files ...') + assert len(source_files) == len(destination_files) + for source_file, destination_file in zip(source_files, destination_files): + self._snapshot_file(source_file, destination_file, class_names, + headers_dict) + + print('Generating and amending gn files ...') + # Ensure that it's no longer used, as it's no longer matched with + # `destination_files` in order after it's sorted below. + source_files = None + + destination_files.sort() + self._create_snapshot_gn_file(destination_files) + + android_gn_pathname = os.path.join(self.destination_project_root_dir, + 'starboard/android/shared/BUILD.gn') + android_gn_content = utils.read_file(android_gn_pathname) + + assert android_gn_content.find( + '"//starboard/android/shared/media_snapshot",') != -1 + + gn_target = ('//starboard/android/shared/media_snapshot/' + + self.media_snapshot_version) + + # Check if there is already a reference + if android_gn_content.find(gn_target) == -1: + android_gn_content = android_gn_content.replace( + '"//starboard/android/shared/media_snapshot",', + '"//starboard/android/shared/media_snapshot",\n "' + gn_target + + '",') + + utils.write_file(android_gn_pathname, android_gn_content) + + self._amend_canonical_sb_implementations() + + def _try_create_canonical_sb_implementation(self): + ''' The canonical Starboard implementation contains files for Sb functions, + e.g. 'player_create.cc' or 'drm_create_system.cc'. They are supposed to + be used without any modifications so they can serve as the reference + during experimentation. + Instead of using the files in the destination folder directly, we make a + copy of these files to `starboard/android/shared/media_snapshot/` + (without any versions), so we can add dispatching code to them, i.e. + + if (GetMediaSnapshotVersion() == 2500) { + return SbPlayerCreate2500(); + } + // Follow the existing implementation + + When a file is copied as canonical implementation, the original file + will be excluded from build in `starboard/android/shared/BUILD.gn` + through `snapshotted_media_files`. + + To simplify implementation, the canonical implementation actually + includes the current implementation using #include directive. + ''' + android_gn_pathname = os.path.join(self.destination_project_root_dir, + 'starboard/android/shared/BUILD.gn') + canonical_snapshot_dir = os.path.join(self.destination_project_root_dir, + self._DESTINATION_SUB_DIR, + 'media_snapshot/') + + android_gn_content = utils.read_file(android_gn_pathname) + + # Sanity check that we are operating on the correct gn file. + assert android_gn_content.find('":starboard_base_symbolize",') != -1 + assert android_gn_content.find('snapshotted_media_files = [') != -1 + + if android_gn_content.find('snapshotted_media_files = []') == -1: + print('Canonical snapshot already created.') + assert os.path.isfile( + os.path.join(canonical_snapshot_dir, 'audio_sink_create.cc')) + return + + print('Creating canonical snapshot ...') + sb_implementation_files = [] + + for target in self._GN_TARGETS: + for pathname in gn_utils.get_source_pathnames( + self.destination_project_root_dir, self._BUILD_CONFIG, target): + if not utils.is_media_file(pathname): + continue + # The canonical implementation is based on the primary implementation + if pathname.find('media_snapshot/') != -1: + continue + + content = utils.read_file(pathname) + if source_utils.is_sb_implementation_file(pathname, content): + sb_implementation_files.append(pathname) + + # We haven't snapshotted yet, `sb_implementation_files` shouldn't be empty + assert len(sb_implementation_files) > 0 + + # Now create the canonical implementation inside + # 'starboard/android/shared/media_snapshot'. Note that any non-canonical + # snapshot will be put into a subfolder of it. + source_file_pathnames = [] + for pathname in sb_implementation_files: + destination_pathname = os.path.join(canonical_snapshot_dir, + os.path.basename(pathname)) + source_file_pathnames.append(destination_pathname) + + if not os.path.isdir(os.path.dirname(destination_pathname)): + os.makedirs(os.path.dirname(destination_pathname)) + + source_utils.create_canonical_file(self.destination_project_root_dir, + pathname, destination_pathname) + + # Create a gn file referring to the above source files + gn_utils.create_gn_file(self.destination_project_root_dir, + os.path.join(canonical_snapshot_dir + 'BUILD.gn'), + 'media_snapshot', source_file_pathnames) + + # Exclude the original source files from 'starboard/android/shared/BUILD.gn' + sb_implementation_files = gn_utils.convert_source_list_to_gn_format( + self.destination_project_root_dir, android_gn_pathname, + sb_implementation_files) + android_gn_content = android_gn_content.replace( + 'snapshotted_media_files = []', 'snapshotted_media_files = [\n ' + + '\n '.join(sb_implementation_files) + '\n ]') + + # Add reference to canonical snapshot + android_gn_content = android_gn_content.replace( + '":starboard_base_symbolize",', + ('":starboard_base_symbolize",\n' + + ' "//starboard/android/shared/media_snapshot",')) + + utils.write_file(android_gn_pathname, android_gn_content) + + def _amend_canonical_sb_implementations(self): + ''' Amend the canonical implementation with branching call to the new + version, e.g. + SbPlayer SbPlayerCreate(...) { + if (GetMediaSnapshotVersion() == 2500) { + return SbPlayerCreate2500(...); + } + // Follow the existing implementation + return ...; + } + + When a file is copied as canonical implementation, the original file + will be excluded from build in `starboard/android/shared/BUILD.gn` + through `snapshotted_media_files`. + ''' + sb_implementation_files = [] + + for pathname in gn_utils.get_source_pathnames( + self.destination_project_root_dir, self._BUILD_CONFIG, + '//starboard/android/shared/media_snapshot'): + if not utils.is_media_file(pathname): + continue + # We only amend the canonical implementation + if pathname.find('media_snapshot/') == -1: + continue + + content = utils.read_file(pathname) + if source_utils.is_sb_implementation_file(pathname, content): + sb_implementation_files.append(pathname) + + assert len(sb_implementation_files) > 0 + if self.verbose: + print('Starboard implementation files', sb_implementation_files) + + for pathname in sb_implementation_files: + with open(pathname, encoding='utf-8') as f: + content = f.read() + content = source_utils.patch_sb_implementation_with_branching_call( + pathname, content, self.media_snapshot_version, self.verbose) + with open(pathname, 'w+', encoding='utf-8') as f: + f.write(content) + + def _snapshot_file(self, source_pathname, destination_pathname, class_names, + headers_dict): + if self.verbose: + print('snapshotting', source_pathname, '=>', destination_pathname) + + with open(source_pathname, encoding='utf-8') as f: + content = f.read() + + if utils.is_header_file(source_pathname): + source_macro = source_utils.generate_include_guard_macro( + self.source_project_root_dir, source_pathname) + destination_macro = source_utils.generate_include_guard_macro( + self.destination_project_root_dir, destination_pathname) + assert content.find(source_macro) > 0 + content = content.replace(source_macro, destination_macro) + assert content.find(source_macro) < 0 + + if not os.path.isdir(os.path.dirname(destination_pathname)): + os.makedirs(os.path.dirname(destination_pathname)) + + for source in headers_dict: + content = content.replace('#include "' + source + '"', + '#include "' + headers_dict[source] + '"') + + content = content.replace('namespace shared', + 'namespace shared_' + self.media_snapshot_version) + + for class_name in class_names: + content = source_utils.replace_class_under_namespace( + content, class_name, 'shared', + 'shared_' + self.media_snapshot_version) + + for symbol_name in [ + 'AudioDurationToFrames', 'AudioFramesToDuration', + 'CanPlayMimeAndKeySystem', 'ErrorCB', 'EndedCB', + 'GetAudioConfiguration', 'GetBytesPerSample', + 'GetMaxVideoInputSizeForCurrentThread', 'IsSDRVideo', 'IsWidevineL1', + 'IsWidevineL3', 'PrerolledCB', 'SetMaxVideoInputSizeForCurrentThread' + ]: + content = source_utils.replace_class_under_namespace( + content, symbol_name, 'shared', + 'shared_' + self.media_snapshot_version) + + content = content.replace( + 'starboard::shared::starboard::player', 'starboard::shared_' + + self.media_snapshot_version + '::starboard::player') + + # The following replacements are very specific. Including here so we don't + # have to modify the generated code manually. + content = content.replace(' ThreadChecker ', + ' shared::starboard::ThreadChecker ') + content = content.replace( + ' Application::Get', + ' ::starboard::shared::starboard::Application::Get') + content = content.replace('worker_ = starboard::make_scoped_ptr(', + 'worker_.reset(') + + content = content.replace('#include "third_party/opus/include', + '#include "third_party/opus/src/include') + + content = source_utils.add_namespace(content, 'JniEnvExt', + '::starboard::android::shared') + content = source_utils.add_namespace(content, 'ScopedJavaByteBuffer', + '::starboard::android::shared') + content = source_utils.add_namespace(content, 'ScopedLocalJavaRef', + '::starboard::android::shared') + + # Replace JNI calls + for jni_prefix in [ + 'Java_dev_cobalt_media_MediaCodecBridge', + 'Java_dev_cobalt_media_MediaDrmBridge' + ]: + content = content.replace(jni_prefix, + jni_prefix + self.media_snapshot_version) + + if source_utils.is_sb_implementation_file(source_pathname, content): + content = source_utils.append_suffix_to_sb_function( + source_pathname, content, self.media_snapshot_version, self.verbose) + + with open(destination_pathname, 'w+', encoding='utf-8') as f: + f.write(content) + + def _create_snapshot_gn_file(self, file_pathnames): + gn_utils.create_gn_file( + self.destination_project_root_dir, + self._get_destination_pathname( + os.path.join(self.source_project_root_dir, 'starboard/BUILD.gn')), + self.media_snapshot_version, file_pathnames) + + +snapshot = Snapshot( + '~/chromium_clean/src', '~/chromium/src', '2500', verbose=False) +snapshot.snapshot() diff --git a/starboard/tools/media/source_utils.py b/starboard/tools/media/source_utils.py new file mode 100644 index 000000000000..c2fd2d879878 --- /dev/null +++ b/starboard/tools/media/source_utils.py @@ -0,0 +1,287 @@ +#!/usr/bin/env python3 +# Copyright 2024 The Cobalt Authors. All Rights Reserved. +# +# 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. +'''Utility functions to manipulate source file.''' + +from datetime import datetime +from textwrap import dedent + +import os +import re +import utils + +_COPYRIGHT_HEADER = '''\ + // Copyright {0} The Cobalt Authors. All Rights Reserved. + // + // 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. + ''' + +_CANONICAL_FILE_CONTENT = '''\ + + {0} + + extern "C" {{ + + SB_EXPORT {1}; + + }} // extern "C" + + {2} {{ + return {3}; + }} + + #define {4} {5} + #define REFERENCE_SOURCE_FILE "{6}" + + #include REFERENCE_SOURCE_FILE + ''' + + +def _get_copyright_header(): + return dedent(_COPYRIGHT_HEADER).format(datetime.now().year) + + +# for `player_create.cc', return 'starboard/player.h'. +def _get_starboard_includes_and_usings(pathname): + basename = os.path.basename(pathname) + + if (basename in [ + 'media_is_audio_supported.cc', 'media_is_supported.cc', + 'media_is_video_supported.cc' + ]): + return dedent('''\ + #include "starboard/shared/starboard/media/media_support_internal.h" + + using starboard::shared::starboard::media::MimeType;''') + + if basename.find('audio_sink_') == 0: + return '#include "starboard/audio_sink.h"' + if basename.find('decode_target_') == 0: + return '#include "starboard/decode_target.h"' + if basename.find('drm_') == 0: + return '#include "starboard/drm.h"' + if basename.find('media_') == 0: + return '#include "starboard/media.h"' + if basename.find('player_') == 0: + return '#include "starboard/player.h"' + assert False + + +# starboard/player.h => STARBOARD_PLAYER_H_ +def generate_include_guard_macro(project_root_dir, file_pathname): + rel_pathname = os.path.relpath(file_pathname, project_root_dir) + return rel_pathname.replace('/', '_').replace('.', '_').upper() + '_' + + +def extract_class_or_struct_names(content): + pattern = r'^\s*class\s+(\w+)\s*[:{]' + matches = re.findall(pattern, content, re.MULTILINE) + + pattern = r'^\s*struct\s+(\w+)\s*[:{]' + return matches + re.findall(pattern, content, re.MULTILINE) + + +def extract_project_includes(content): + pattern = r'#\s*include\s*"([^"]+)"' + return re.findall(pattern, content, re.MULTILINE) + + +# This replace 'JNIEnv' to the content of new_text, but won't replace +# 'shared::JNIEnv'. +def replace_class_name_without_namespace(content, class_name, new_text): + pattern = r'(? 0 + if namespace[-2:] != '::': + namespace += '::' + return replace_class_name_without_namespace(content, class_name, + namespace + class_name) + + +def replace_class_under_namespace(content, class_name, old_namespace, + new_namespace): + pattern = rf'{old_namespace}::(\S*)?{class_name}' + replacement = rf'{new_namespace}::\1{class_name}' + return re.sub(pattern, replacement, content) + + +def is_header_file(file_pathname): + return file_pathname[-2:] == '.h' + + +def add_header_file(content, header_file): + is_project_header = header_file.find('/') >= 0 + + if is_project_header: + include_directive = '#include "' + header_file + '"' + include_directive_prefix = '#include "' + search_func = str.rfind + else: + include_directive = '#include <' + header_file + '>' + include_directive_prefix = '#include <' + if header_file.find('.'): + search_func = str.find # C header files first + else: + search_func = str.rfind # C++ header files second + + if content.find(include_directive) >= 0: + # Already included + return content + + index = search_func(content, include_directive_prefix) + if index >= 0: + index = content.find('\n', index) + return content[:index] + '\n' + include_directive + '\n' + content[index:] + + # Cannot find the group, add it to the very beginning and refine it manually + return include_directive + '\n' + content + + +def is_sb_implementation_file(pathname, content): + basename, ext = utils.get_base_file_name_and_ext(pathname) + if ext != 'cc': + return False + + function_name = utils.base_name_to_sb_function_name(basename) + if content.find(' ' + function_name + '(') == -1: + return False + return True + + +def append_suffix_to_sb_function(pathname, content, suffix, verbose): + if verbose: + print('appending suffix for Sb function to', pathname) + + basename, ext = utils.get_base_file_name_and_ext(pathname) + assert ext == 'cc' + + function_name = utils.base_name_to_sb_function_name(basename) + + assert content.find(' ' + function_name + '(') != -1 + + return content.replace(function_name, function_name + suffix) + + +# For the content of player_destroy.cc and 'SbPlayerDestroy', returns +# 'void SbPlayerDestroy(SbPlayer player)'. +def get_function_prototype(content, function_name): + # We assume the first occurrence is the prototype. This won't work when + # there is a comment before the prototype, which isn't currently used. + left = content.find(' ' + function_name + '(') + + # Move to right after the previous line end + left = content.rfind('\n', 0, left) + 1 + assert left > 0 + + # Now keep find until the next ')' + right = content.find(')', left) + assert right != -1 + + return content[left:right + 1] + + +def patch_sb_implementation_with_branching_call(pathname, content, suffix, + verbose): + if verbose: + print('patching', pathname) + + basename, ext = utils.get_base_file_name_and_ext(pathname) + assert ext == 'cc' + + function_name = utils.base_name_to_sb_function_name(basename) + + if content.find(function_name + suffix) != -1: + # Already patched, don't patch further. + return content + + assert content.find(' ' + function_name + '(') != -1 + + prototype = get_function_prototype(content, function_name) + + branched_prototype = prototype.replace(function_name, function_name + suffix) + + content = add_header_file(content, + 'starboard/shared/media_snapshot/media_snapshot.h') + + offset = content.find(prototype) + content = content[:offset] + branched_prototype + ';\n\n' + content[offset:] + + offset = content.find(prototype) + offset = content.find('{', offset) + assert offset != -1 + offset += 1 + + calling_statement = utils.get_calling_statement_from_prototype( + branched_prototype) + if_statement = ('\n' + ' if (GetMediaSnapshotVersion() == ' + suffix + + ') {\n' + ' return ' + calling_statement + ';\n' + ' }\n') + + return content[:offset] + if_statement + content[offset:] + + +def create_canonical_file(source_project_root_dir, source_pathname, + destination_pathname): + assert os.path.basename(source_pathname) == os.path.basename( + destination_pathname) + + with open(source_pathname, encoding='utf-8') as f: + content = f.read() + + basename, ext = utils.get_base_file_name_and_ext(source_pathname) + assert ext == 'cc' + + # '.../decode_target_release.cc' => SbDecodeTargetRelease + function_name = utils.base_name_to_sb_function_name(basename) + + assert content.find(' ' + function_name + '(') != -1 + + # SbDecodeTargetRelease => starboard/decode_target.h + includes_and_usings = _get_starboard_includes_and_usings(source_pathname) + + # SbDecodeTargetRelease => + # void SbDecodeTargetRelease(SbDecodeTarget decode_target) + prototype = get_function_prototype(content, function_name) + + canonical_function_name = function_name + 'Canonical' + + # void SbDecodeTargetRelease(SbDecodeTarget decode_target) => + # void SbDecodeTargetReleaseCanonical(SbDecodeTarget decode_target) + canonical_prototype = prototype.replace(function_name, + canonical_function_name) + + # void SbDecodeTargetReleaseCanonical(SbDecodeTarget decode_target) => + # SbDecodeTargetReleaseCanonical(decode_target) + calling_statement = utils.get_calling_statement_from_prototype( + canonical_prototype) + + source_rel_pathname = os.path.relpath(source_pathname, + source_project_root_dir) + with open(destination_pathname, 'w+', encoding='utf-8') as f: + f.write(_get_copyright_header() + dedent(_CANONICAL_FILE_CONTENT).format( + includes_and_usings, canonical_prototype, prototype, calling_statement, + function_name, canonical_function_name, source_rel_pathname)) diff --git a/starboard/tools/media/utils.py b/starboard/tools/media/utils.py new file mode 100644 index 000000000000..ac270494271f --- /dev/null +++ b/starboard/tools/media/utils.py @@ -0,0 +1,112 @@ +#!/usr/bin/env python3 +# Copyright 2024 The Cobalt Authors. All Rights Reserved. +# +# 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. +"""Utility functions.""" + +import os + + +def is_media_file(pathname): + # Image decoder and media session are not media + if pathname.find('image') >= 0 or pathname.find('media_session') >= 0: + return False + + # While these two are media files, they are unused on Android TV and cause + # build errors. + if pathname.find('media_get_buffer_storage_type.cc') >= 0: + return False + if pathname.find('punchout_video_renderer_sink.cc') >= 0: + return False + + # These files should be the same among media implementations, and should be + # manually consolidated if there are any differences + # TODO: Print them to log if there are any differences + if (pathname.find('player_create.cc') >= 0 or + pathname.find('player_destroy.cc') >= 0): + return False + + # Assume the function in starboard/common are the same across all versions. + # Exclude them to avoid the extra handling required as they are under the root + # starboard namespace and the functions are called directly without any + # namespace specifier. + if pathname.find('starboard/common/') >= 0: + return False + + if (pathname.find('starboard/shared/starboard/audio_sink') >= 0 or + pathname.find('starboard/shared/starboard/decode_target') >= 0 or + pathname.find('starboard/shared/starboard/drm') >= 0 or + pathname.find('starboard/shared/starboard/media') >= 0 or + pathname.find('starboard/shared/starboard/opus') >= 0 or + pathname.find('starboard/shared/starboard/player') >= 0 or + pathname.find('starboard/shared/starboard/drm') >= 0): + return True + + return (pathname.find('audio') >= 0 or pathname.find('decode') >= 0 or + pathname.find('drm') >= 0 or pathname.find('media') >= 0 or + pathname.find('player') >= 0 or pathname.find('video') >= 0) + + +def is_header_file(pathname): + return pathname[-2:] == '.h' + + +# '.../starboard/android/shared/player_create.cc' => 'player_create', 'cc' +def get_base_file_name_and_ext(pathname): + # '.../starboard/android/shared/player_create.cc' => player_create.cc + basename = os.path.basename(pathname) + + return basename.split('.') + + +def base_name_to_sb_function_name(pathname): + # 'player_create' => 'SbPlayerCreate' + return 'Sb' + ''.join(x.capitalize() for x in pathname.split('_')) + + +# For prototype +# const void* SbDrmGetMetrics(SbDrmSystem drm_system, int* size) +# returns +# SbDrmGetMetrics(drm_system, size) +# i.e. remove the return type, and all types of parameters. +def get_calling_statement_from_prototype(prototype): + # multiline to single line + calling_statement = prototype.replace('\n', '') + calling_statement = calling_statement.strip() + + # const void* SbDrmGetMetrics(SbDrmSystem drm_system, int* size) => + # SbDrmGetMetrics(SbDrmSystem drm_system, int* size) + assert calling_statement.find(' Sb') != -1 + calling_statement = calling_statement[calling_statement.find(' Sb') + 1:] + + # SbDrmGetMetrics(SbDrmSystem drm_system, int* size) => + # SbDrmGetMetrics(drm_system, size) + assert calling_statement[-1] == ')' + calling_statement = calling_statement[:-1] + name, parameters = calling_statement.split('(') + + arguments = [] + for parameter in parameters.split(','): + arguments.append(parameter.split(' ')[-1]) + + return name + '(' + ', '.join(arguments) + ')' + + +def read_file(pathname): + with open(pathname, encoding='utf-8') as f: + return f.read() + + +def write_file(pathname, content): + with open(pathname, 'w+', encoding='utf-8') as f: + f.write(content)