"""Utilities for mapping browser versions to webassembly features."""
import logging
from enum import IntEnum, auto
from . import diagnostics
from .settings import settings, user_settings
logger = logging.getLogger('feature_matrix')
UNSUPPORTED = 0x7FFFFFFF
OLDEST_SUPPORTED_CHROME = 74
OLDEST_SUPPORTED_FIREFOX = 68
OLDEST_SUPPORTED_SAFARI = 120200
OLDEST_SUPPORTED_NODE = 122209
class Feature(IntEnum):
MUTABLE_GLOBALS = auto()
NON_TRAPPING_FPTOINT = auto()
SIGN_EXT = auto()
BULK_MEMORY = auto()
JS_BIGINT_INTEGRATION = auto()
THREADS = auto()
PROMISE_ANY = auto()
MEMORY64 = auto()
WORKER_ES6_MODULES = auto()
OFFSCREENCANVAS_SUPPORT = auto()
WASM_LEGACY_EXCEPTIONS = auto()
WASM_EXCEPTIONS = auto()
WEBGL2 = auto()
WEBGPU = auto()
GROWABLE_ARRAYBUFFERS = auto()
disable_override_features = set()
enable_override_features = set()
min_browser_versions = {
Feature.MUTABLE_GLOBALS: {
'chrome': 74,
'firefox': 61,
'safari': 130100,
'node': 120000,
},
Feature.NON_TRAPPING_FPTOINT: {
'chrome': 75,
'firefox': 65,
'safari': 150000,
'node': 130000,
},
Feature.SIGN_EXT: {
'chrome': 74,
'firefox': 62,
'safari': 140100,
'node': 120000,
},
Feature.BULK_MEMORY: {
'chrome': 75,
'firefox': 79,
'safari': 150000,
'node': 130000,
},
Feature.JS_BIGINT_INTEGRATION: {
'chrome': 67,
'firefox': 78,
'safari': 150000,
'node': 130000,
},
Feature.THREADS: {
'chrome': 74,
'firefox': 79,
'safari': 140100,
'node': 160400,
},
Feature.PROMISE_ANY: {
'chrome': 85,
'firefox': 79,
'safari': 140000,
'node': 150000,
},
Feature.MEMORY64: {
'chrome': 128,
'firefox': 129,
'safari': UNSUPPORTED,
'node': 230000,
},
Feature.WEBGL2: {
'chrome': 56,
'firefox': 51,
'safari': 150000,
'node': UNSUPPORTED,
},
Feature.WEBGPU: {
'chrome': 113,
'firefox': 141,
'safari': 260000,
'node': UNSUPPORTED,
},
Feature.WORKER_ES6_MODULES: {
'chrome': 80,
'firefox': 114,
'safari': 150000,
'node': 0,
},
Feature.OFFSCREENCANVAS_SUPPORT: {
'chrome': 69,
'firefox': 105,
'safari': 170000,
'node': 0,
},
Feature.WASM_LEGACY_EXCEPTIONS: {
'chrome': 95,
'firefox': 100,
'safari': 150200,
'node': 170000,
},
Feature.WASM_EXCEPTIONS: {
'chrome': 137,
'firefox': 131,
'safari': 180400,
'node': 220000,
},
Feature.GROWABLE_ARRAYBUFFERS: {
'chrome': 136,
'firefox': 145,
'safari': UNSUPPORTED,
'node': 240000,
},
}
for feature, reqs in min_browser_versions.items():
always_present = (reqs['chrome'] <= OLDEST_SUPPORTED_CHROME and
reqs['firefox'] <= OLDEST_SUPPORTED_FIREFOX and
reqs['safari'] <= OLDEST_SUPPORTED_SAFARI and
reqs['node'] <= OLDEST_SUPPORTED_NODE)
assert not always_present, f'{feature.name} is no longer needed'
def caniuse(feature):
if feature in disable_override_features:
return False
if feature in enable_override_features:
return True
min_versions = min_browser_versions[feature]
def report_missing(setting_name):
setting_value = getattr(settings, setting_name)
logger.debug(f'cannot use {feature.name} because {setting_name} is too old: {setting_value}')
if settings.MIN_CHROME_VERSION < min_versions['chrome']:
report_missing('MIN_CHROME_VERSION')
return False
if settings.MIN_FIREFOX_VERSION < min_versions['firefox']:
report_missing('MIN_FIREFOX_VERSION')
return False
if settings.MIN_SAFARI_VERSION < min_versions['safari']:
report_missing('MIN_SAFARI_VERSION')
return False
if 'node' in min_versions and settings.MIN_NODE_VERSION < min_versions['node']:
report_missing('MIN_NODE_VERSION')
return False
return True
def enable_feature(feature, reason, override=False):
"""Updates default settings for browser versions such that the given
feature is available everywhere.
"""
if override:
enable_override_features.add(feature)
for name, min_version in min_browser_versions[feature].items():
name = f'MIN_{name.upper()}_VERSION'
if settings[name] < min_version:
if name in user_settings:
if name == 'MIN_SAFARI_VERSION' and reason == 'pthreads':
continue
diagnostics.warning(
'compatibility',
f'{name}={user_settings[name]} is not compatible with {reason} '
f'({name}={min_version} or above required)')
else:
logger.debug(f'Enabling {name}={min_version} to accommodate {reason}')
setattr(settings, name, min_version)
def disable_feature(feature):
"""Allow the user to disable a feature that would otherwise be on by default.
"""
disable_override_features.add(feature)
def apply_min_browser_versions():
if settings.WASM_BIGINT and 'WASM_BIGINT' in user_settings:
enable_feature(Feature.JS_BIGINT_INTEGRATION, 'WASM_BIGINT')
if settings.PTHREADS:
enable_feature(Feature.THREADS, 'pthreads')
enable_feature(Feature.BULK_MEMORY, 'pthreads')
elif settings.WASM_WORKERS or settings.SHARED_MEMORY:
enable_feature(Feature.BULK_MEMORY, 'shared-mem')
if settings.RELOCATABLE:
enable_feature(Feature.MUTABLE_GLOBALS, 'dynamic linking')
if settings.MEMORY64 == 1:
enable_feature(Feature.MEMORY64, 'MEMORY64')
if settings.EXPORT_ES6 and settings.PTHREADS:
enable_feature(Feature.WORKER_ES6_MODULES, 'EXPORT_ES6 with -pthread')
if settings.EXPORT_ES6 and settings.WASM_WORKERS:
enable_feature(Feature.WORKER_ES6_MODULES, 'EXPORT_ES6 with -sWASM_WORKERS')
if settings.OFFSCREENCANVAS_SUPPORT:
enable_feature(Feature.OFFSCREENCANVAS_SUPPORT, 'OFFSCREENCANVAS_SUPPORT')
if settings.WASM_EXCEPTIONS or settings.SUPPORT_LONGJMP == 'wasm':
if settings.WASM_LEGACY_EXCEPTIONS:
enable_feature(Feature.WASM_LEGACY_EXCEPTIONS, 'Wasm Legacy exceptions (-fwasm-exceptions with -sWASM_LEGACY_EXCEPTIONS=1)')
else:
enable_feature(Feature.WASM_EXCEPTIONS, 'Wasm exceptions (-fwasm-exceptions with -sWASM_LEGACY_EXCEPTIONS=0)')