mirror of
https://bitbucket.org/chromiumembedded/cef
synced 2025-04-03 13:41:09 +02:00
- Generated files are now created when running cef_create_projects or the new version_manager.py tool. These files are still created in the cef/ source tree (same location as before) but Git ignores them due to the generated .gitignore file. - API hashes are committed to Git as a new cef_api_versions.json file. This file is used for both code generation and CEF version calculation (replacing the previous usage of cef_api_hash.h for this purpose). It will be updated by the CEF admin before merging breaking API changes upstream. - As an added benefit to the above, contributor PRs will no longer contain generated code that is susceptible to frequent merge conflicts. - From a code generation perspective, the main difference is that we now use versioned structs (e.g. cef_browser_0_t instead of cef_browser_t) on the libcef (dll/framework) side. Most of the make_*.py tool changes are related to supporting this. - From the client perspective, you can now define CEF_API_VERSION in the project configuration (or get CEF_EXPERIMENTAL by default). This define will change the API exposed in CEF’s include/ and include/capi header files. All client-side targets including libcef_dll_wrapper will need be recompiled when changing this define. - Examples of the new API-related define usage are provided in cef_api_version_test.h, api_version_test_impl.cc and api_version_unittest.cc. To test: - Run `ceftests --gtest_filter=ApiVersionTest.*` - Add `cef_api_version=13300` to GN_DEFINES. Re-run configure, build and ceftests steps. - Repeat with 13301, 13302, 13303 (all supported test versions).
219 lines
5.7 KiB
Python
219 lines
5.7 KiB
Python
# Copyright (c) 2018 The Chromium Embedded Framework Authors. All rights
|
|
# reserved. Use of this source code is governed by a BSD-style license that
|
|
# can be found in the LICENSE file.
|
|
|
|
from __future__ import absolute_import
|
|
from cef_parser import *
|
|
from file_util import *
|
|
import os
|
|
|
|
# Other headers that export C API functions.
|
|
OTHER_HEADERS = [
|
|
'cef_api_hash.h',
|
|
'cef_id_mappers.h',
|
|
'cef_version_info.h',
|
|
'internal/cef_dump_without_crashing_internal.h',
|
|
'internal/cef_logging_internal.h',
|
|
'internal/cef_string_list.h',
|
|
'internal/cef_string_map.h',
|
|
'internal/cef_string_multimap.h',
|
|
'internal/cef_string_types.h',
|
|
'internal/cef_thread_internal.h',
|
|
'internal/cef_time.h',
|
|
'internal/cef_trace_event_internal.h',
|
|
]
|
|
|
|
|
|
def make_libcef_dll_dylib_impl_parts(name, retval, args):
|
|
# Split arguments into types and names.
|
|
arg_types = ''
|
|
arg_names = ''
|
|
for arg in args:
|
|
if len(arg_types) > 0:
|
|
arg_types += ', '
|
|
arg_names += ', '
|
|
pos = arg.rfind(' ')
|
|
arg_types += arg[0:pos]
|
|
arg_names += arg[pos + 1:]
|
|
|
|
declare = 'decltype(&%s) %s;\n' % (name, name)
|
|
|
|
init = 'INIT_ENTRY(%s);\n' % name
|
|
|
|
impl = """NO_SANITIZE("cfi-icall") %s %s(%s) {
|
|
%sg_libcef_pointers.%s(%s);
|
|
}
|
|
""" % (retval, name, ', '.join(args), 'return '
|
|
if retval != 'void' else '', name, arg_names)
|
|
|
|
return (declare, init, impl)
|
|
|
|
|
|
def make_libcef_dll_dylib_impl_func(func):
|
|
name = func.get_capi_name()
|
|
parts = func.get_capi_parts([], True)
|
|
retval = parts['retval']
|
|
args = parts['args']
|
|
return make_libcef_dll_dylib_impl_parts(name, retval, args)
|
|
|
|
|
|
def make_libcef_dll_dylib_impl(header):
|
|
filenames = []
|
|
includes = [
|
|
'#include "include/base/cef_compiler_specific.h"',
|
|
'#include "include/wrapper/cef_library_loader.h"',
|
|
]
|
|
ptr_declare = ''
|
|
ptr_init = ''
|
|
ptr_impl = ''
|
|
|
|
# Include required headers for global functions.
|
|
for func in header.get_funcs():
|
|
declare, init, impl = make_libcef_dll_dylib_impl_func(func)
|
|
pre, post = get_version_surround(func)
|
|
ptr_declare += pre + declare + post
|
|
ptr_init += pre + init + post
|
|
ptr_impl += pre + impl + post + '\n'
|
|
|
|
filename = func.get_file_name()
|
|
if not filename in filenames:
|
|
includes.append('#include "include/capi/%s"' % func.get_capi_file_name())
|
|
filenames.append(filename)
|
|
|
|
# Include required headers for static class functions.
|
|
allclasses = header.get_classes()
|
|
for cls in allclasses:
|
|
funcs = cls.get_static_funcs()
|
|
for func in funcs:
|
|
declare, init, impl = make_libcef_dll_dylib_impl_func(func)
|
|
pre1, post1 = get_version_surround(func)
|
|
pre2, post2 = get_version_surround(cls)
|
|
pre = pre1 + pre2
|
|
post = post1 + post2
|
|
ptr_declare += pre + declare + post
|
|
ptr_init += pre + init + post
|
|
ptr_impl += pre + impl + post + '\n'
|
|
|
|
if len(funcs) > 0:
|
|
filename = cls.get_file_name()
|
|
if not filename in filenames:
|
|
includes.append('#include "include/capi/%s"' % cls.get_capi_file_name())
|
|
filenames.append(filename)
|
|
|
|
# Parse other headers.
|
|
root_directory = header.get_root_directory()
|
|
for other in OTHER_HEADERS:
|
|
path = os.path.join(root_directory, other)
|
|
content = read_file(path)
|
|
funcs = get_function_impls(content, 'CEF_EXPORT', False)
|
|
for func in funcs:
|
|
declare, init, impl = make_libcef_dll_dylib_impl_parts(
|
|
func['name'], func['retval'], func['args'])
|
|
ptr_declare += declare
|
|
ptr_init += init
|
|
ptr_impl += impl + '\n'
|
|
|
|
includes.append('#include "include/%s"' % other)
|
|
|
|
# Build the final output.
|
|
result = get_copyright() + """
|
|
|
|
#include <dlfcn.h>
|
|
#include <stdio.h>
|
|
|
|
""" + "\n".join(sorted(includes)) + """
|
|
|
|
// GLOBAL WRAPPER FUNCTIONS - Do not edit by hand.
|
|
|
|
namespace {
|
|
|
|
void* g_libcef_handle = nullptr;
|
|
|
|
void* libcef_get_ptr(const char* path, const char* name) {
|
|
void* ptr = dlsym(g_libcef_handle, name);
|
|
if (!ptr) {
|
|
fprintf(stderr, "dlsym %s: %s\\n", path, dlerror());
|
|
}
|
|
return ptr;
|
|
}
|
|
|
|
struct libcef_pointers {
|
|
""" + ptr_declare + """
|
|
} g_libcef_pointers = {0};
|
|
|
|
#define INIT_ENTRY(name) \
|
|
g_libcef_pointers.name = (decltype(&name))libcef_get_ptr(path, #name); \
|
|
if (!g_libcef_pointers.name) { \
|
|
return 0; \
|
|
}
|
|
|
|
int libcef_init_pointers(const char* path) {
|
|
""" + ptr_init + """
|
|
return 1;
|
|
}
|
|
|
|
} // namespace
|
|
|
|
int cef_load_library(const char* path) {
|
|
if (g_libcef_handle)
|
|
return 0;
|
|
|
|
g_libcef_handle = dlopen(path, RTLD_LAZY | RTLD_LOCAL | RTLD_FIRST);
|
|
if (!g_libcef_handle) {
|
|
fprintf(stderr, "dlopen %s: %s\\n", path, dlerror());
|
|
return 0;
|
|
}
|
|
|
|
if (!libcef_init_pointers(path)) {
|
|
cef_unload_library();
|
|
return 0;
|
|
}
|
|
|
|
return 1;
|
|
}
|
|
|
|
int cef_unload_library() {
|
|
int result = 0;
|
|
if (g_libcef_handle) {
|
|
result = !dlclose(g_libcef_handle);
|
|
if (!result) {
|
|
fprintf(stderr, "dlclose: %s\\n", dlerror());
|
|
}
|
|
g_libcef_handle = nullptr;
|
|
}
|
|
return result;
|
|
}
|
|
|
|
""" + ptr_impl
|
|
return result
|
|
|
|
|
|
def write_libcef_dll_dylib_impl(header, file):
|
|
newcontents = make_libcef_dll_dylib_impl(header)
|
|
return (file, newcontents)
|
|
|
|
|
|
# Test the module.
|
|
if __name__ == "__main__":
|
|
import sys
|
|
|
|
# Verify that the correct number of command-line arguments are provided.
|
|
if len(sys.argv) < 2:
|
|
sys.stderr.write('Usage: ' + sys.argv[0] + ' <cpp_header_dir>\n')
|
|
sys.exit()
|
|
|
|
cpp_header_dir = sys.argv[1]
|
|
|
|
# Create the header object. Should match the logic in translator.py.
|
|
header = obj_header()
|
|
header.set_root_directory(cpp_header_dir)
|
|
excluded_files = [
|
|
'cef_api_hash.h', 'cef_application_mac.h', 'cef_version_info.h'
|
|
]
|
|
header.add_directory(cpp_header_dir, excluded_files)
|
|
header.add_directory(os.path.join(cpp_header_dir, 'test'))
|
|
header.add_directory(os.path.join(cpp_header_dir, 'views'))
|
|
|
|
# Dump the result to stdout.
|
|
sys.stdout.write(make_libcef_dll_dylib_impl(header))
|