#!/usr/bin/env python
# xpidl.py - A parser for cross-platform IDL (XPIDL) files.
#
# This Source Code Form is subject to the terms of the Mozilla Public
# License, v. 2.0. If a copy of the MPL was not distributed with this
# file, You can obtain one at http://mozilla.org/MPL/2.0/.
"""A parser for cross-platform IDL (XPIDL) files."""
import os.path
import re
import sys
import textwrap
from collections
import namedtuple
import six
from ply
import lex, yacc
"""A type conforms to the following pattern:
def nativeType(self, calltype):
'returns a string representation of the native type
calltype must be
'in',
'out',
'inout',
or 'element'
Interface members const/method/attribute conform to the following pattern:
name =
'string'
def toIDL(self):
'returns the member signature as IDL'
"""
# XXX(nika): Fix the IDL files which do this so we can remove this list?
def rustPreventForward(s):
"""These types are foward declared as interfaces, but never actually defined
in IDL files. We don
't want to generate references to them in rust for that
reason.
"""
return s
in (
"nsIFrame",
"nsSubDocumentFrame",
)
def attlistToIDL(attlist):
if len(attlist) == 0:
return ""
attlist = list(attlist)
attlist.sort(key=
lambda a: a[0])
return "[%s] " %
",".join(
[
"%s%s" % (name, value
is not None and "(%s)" % value
or "")
for name, value, aloc
in attlist
]
)
_paramsHardcode = {
2: (
"array",
"shared",
"iid_is",
"size_is",
"retval"),
3: (
"array",
"size_is",
"const"),
}
def paramAttlistToIDL(attlist):
if len(attlist) == 0:
return ""
# Hack alert: g_hash_table_foreach is pretty much unimitatable... hardcode
# quirk
attlist = list(attlist)
sorted = []
if len(attlist)
in _paramsHardcode:
for p
in _paramsHardcode[len(attlist)]:
i = 0
while i < len(attlist):
if attlist[i][0] == p:
sorted.append(attlist[i])
del attlist[i]
continue
i += 1
sorted.extend(attlist)
return "[%s] " %
", ".join(
[
"%s%s" % (name, value
is not None and " (%s)" % value
or "")
for name, value, aloc
in sorted
]
)
def unaliasType(t):
while t.kind ==
"typedef":
t = t.realtype
assert t
is not None
return t
def getBuiltinOrNativeTypeName(t):
t = unaliasType(t)
if t.kind ==
"builtin":
return t.name
elif t.kind ==
"native":
assert t.specialtype
is not None
return "[%s]" % t.specialtype
else:
return None
class BuiltinLocation(object):
def get(self):
return ""
def __str__(self):
return self.get()
class Builtin(object):
kind =
"builtin"
location = BuiltinLocation
def __init__(
self, name, nativename, rustname, tsname, signed=
False, maybeConst=
False
):
self.name = name
self.nativename = nativename
self.rustname = rustname
self.tsname = tsname
self.signed = signed
self.maybeConst = maybeConst
def isPointer(self):
"""Check if this type is a pointer type - this will control how pointers act"""
return self.nativename.endswith(
"*")
def nativeType(self, calltype, shared=
False, const=
False):
if self.name
in [
"string",
"wstring"]
and calltype ==
"element":
raise IDLError(
"Use string class types for string Array elements", self.location
)
if const:
print(
IDLError(
"[const] doesn't make sense on builtin types.",
self.location,
warning=
True,
),
file=sys.stderr,
)
const =
"const "
elif calltype ==
"in" and self.isPointer():
const =
"const "
elif shared:
if not self.isPointer():
raise IDLError(
"[shared] not applicable to non-pointer types.", self.location
)
const =
"const "
else:
const =
""
return "%s%s %s" % (const, self.nativename,
"*" if "out" in calltype
else "")
def rustType(self, calltype, shared=
False, const=
False):
# We want to rewrite any *mut pointers to *const pointers if constness
# was requested.
const = const
or (
"out" not in calltype
and self.isPointer())
or shared
rustname = self.rustname
if const
and self.isPointer():
rustname = self.rustname.replace(
"*mut",
"*const")
return "%s%s" % (
"*mut " if "out" in calltype
else "", rustname)
def tsType(self):
if self.tsname:
return self.tsname
raise TSNoncompat(f
"Builtin type {self.name} unsupported in TypeScript")
builtinNames = [
Builtin(
"boolean",
"bool",
"bool",
"boolean"),
Builtin(
"void",
"void",
"libc::c_void",
"void"),
Builtin(
"int8_t",
"int8_t",
"i8",
"i8",
True,
True),
Builtin(
"int16_t",
"int16_t",
"i16",
"i16",
True,
True),
Builtin(
"int32_t",
"int32_t",
"i32",
"i32",
True,
True),
Builtin(
"int64_t",
"int64_t",
"i64",
"i64",
True,
True),
Builtin(
"uint8_t",
"uint8_t",
"u8",
"u8",
False,
True),
Builtin(
"uint16_t",
"uint16_t",
"u16",
"u16",
False,
True),
Builtin(
"uint32_t",
"uint32_t",
"u32",
"u32",
False,
True),
Builtin(
"uint64_t",
"uint64_t",
"u64",
"u64",
False,
True),
Builtin(
"nsresult",
"nsresult",
"nserror::nsresult",
"nsresult"),
Builtin(
"float",
"float",
"libc::c_float",
"float"),
Builtin(
"double",
"double",
"libc::c_double",
"double"),
Builtin(
"char",
"char",
"libc::c_char",
"string"),
Builtin(
"string",
"char *",
"*const libc::c_char",
"string"),
Builtin(
"wchar",
"char16_t",
"u16",
"string"),
Builtin(
"wstring",
"char16_t *",
"*const u16",
"string"),
# NOTE: char16_t is the same type as `wchar` in C++, however it is reflected
# into JS as an integer, allowing it to be used in constants.
# This inconsistency sucks, but reflects existing usage so unfortunately
# isn't very easy to change.
Builtin(
"char16_t",
"char16_t",
"u16",
"u16",
False,
True),
# As seen in mfbt/RefCountType.h, this type has special handling to
# maintain binary compatibility with MSCOM's IUnknown that cannot be
# expressed in XPIDL.
Builtin(
"MozExternalRefCountType",
"MozExternalRefCountType",
"MozExternalRefCountType",
None,
),
]
# Allow using more C-style names for the basic integer types.
builtinAlias = [
(
"octet",
"uint8_t"),
(
"unsigned short",
"uint16_t"),
(
"unsigned long",
"uint32_t"),
(
"unsigned long long",
"uint64_t"),
(
"short",
"int16_t"),
(
"long",
"int32_t"),
(
"long long",
"int64_t"),
]
builtinMap = {}
for b
in builtinNames:
builtinMap[b.name] = b
for alias, name
in builtinAlias:
builtinMap[alias] = builtinMap[name]
class Location(object):
_line =
None
def __init__(self, lexer, lineno, lexpos):
self._lineno = lineno
self._lexpos = lexpos
self._lexdata = lexer.lexdata
self._file = getattr(lexer,
"filename",
"")
def __eq__(self, other):
return self._lexpos == other._lexpos
and self._file == other._file
def resolve(self):
if self._line:
return
startofline = self._lexdata.rfind(
"\n", 0, self._lexpos) + 1
endofline = self._lexdata.find(
"\n", self._lexpos, self._lexpos + 80)
self._line = self._lexdata[startofline:endofline]
self._colno = self._lexpos - startofline
def pointerline(self):
def i():
for i
in range(0, self._colno):
yield " "
yield "^"
return "".join(i())
def get(self):
self.resolve()
return "%s line %s:%s" % (self._file, self._lineno, self._colno)
def __str__(self):
self.resolve()
return "%s line %s:%s\n%s\n%s" % (
self._file,
self._lineno,
self._colno,
self._line,
self.pointerline(),
)
class NameMap(object):
"""Map of name -> object. Each object must have a .name and .location property.
Setting the same name twice throws an error.
"""
def __init__(self):
self._d = {}
def __getitem__(self, key):
if key
in builtinMap:
return builtinMap[key]
return self._d[key]
def __iter__(self):
return six.itervalues(self._d)
def __contains__(self, key):
return key
in builtinMap
or key
in self._d
def set(self, object):
if object.name
in builtinMap:
raise IDLError(
"name '%s' is a builtin and cannot be redeclared" % (object.name),
object.location,
)
if object.name.startswith(
"_"):
object.name = object.name[1:]
if object.name
in self._d:
old = self._d[object.name]
if old == object:
return
if isinstance(old, Forward)
and isinstance(object, Interface):
self._d[object.name] = object
elif isinstance(old, Interface)
and isinstance(object, Forward):
pass
else:
raise IDLError(
"name '%s' specified twice. Previous location: %s"
% (object.name, self._d[object.name].location),
object.location,
)
else:
self._d[object.name] = object
def get(self, id, location):
try:
return self[id]
except KeyError:
raise IDLError(f
"Name '{id}' not found", location)
class RustNoncompat(Exception):
"""
This exception
is raised when a particular type
or function cannot be safely exposed to
rust code
"""
def __init__(self, reason):
self.reason = reason
def __str__(self):
return self.reason
class TSNoncompat(Exception):
"""Raised when a type cannot be exposed to TypeScript."""
def __init__(self, reason):
self.reason = reason
def __str__(self):
return self.reason
class IDLError(Exception):
def __init__(self, message, location, warning=
False, notes=
None):
self.message = message
self.location = location
self.warning = warning
self.notes = notes
def __str__(self):
error =
"%s: %s, %s" % (
self.warning
and "warning" or "error",
self.message,
self.location,
)
if self.notes
is not None:
error +=
"\nnote: %s" % self.notes
return error
class Include(object):
kind =
"include"
def __init__(self, filename, location):
self.filename = filename
self.location = location
def __str__(self):
return "".join([
"include '%s'\n" % self.filename])
def resolve(self, parent):
def incfiles():
yield self.filename
for dir
in parent.incdirs:
yield os.path.join(dir, self.filename)
for file
in incfiles():
if not os.path.exists(file):
continue
if file
in parent.includeCache:
self.IDL = parent.includeCache[file]
else:
self.IDL = parent.parser.parse(
open(file, encoding=
"utf-8").read(), filename=file
)
self.IDL.resolve(
parent.incdirs,
parent.parser,
parent.webidlconfig,
parent.includeCache,
)
parent.includeCache[file] = self.IDL
for type
in self.IDL.getNames():
parent.setName(type)
parent.deps.extend(self.IDL.deps)
return
raise IDLError(
"File '%s' not found" % self.filename, self.location)
class IDL(object):
def __init__(self, productions):
self.hasSequence =
False
self.productions = productions
self.deps = []
def setName(self, object):
self.namemap.set(object)
def getName(self, id, location):
if id.name ==
"Array":
if id.params
is None or len(id.params) != 1:
raise IDLError(
"Array takes exactly 1 parameter", location)
self.hasSequence =
True
return Array(self.getName(id.params[0], location), location)
if id.params
is not None:
raise IDLError(
"Generic type '%s' unrecognized" % id.name, location)
try:
return self.namemap[id.name]
except KeyError:
raise IDLError(
"type '%s' not found" % id.name, location)
def hasName(self, id):
return id
in self.namemap
def getNames(self):
return iter(self.namemap)
def __str__(self):
return "".join([str(p)
for p
in self.productions])
def resolve(self, incdirs, parser, webidlconfig, includeCache=
None):
self.namemap = NameMap()
self.incdirs = incdirs
self.parser = parser
self.webidlconfig = webidlconfig
self.includeCache = {}
if includeCache
is None else includeCache
for p
in self.productions:
p.resolve(self)
def includes(self):
for p
in self.productions:
if p.kind ==
"include":
yield p
if self.hasSequence:
yield Include(
"nsTArray.h", BuiltinLocation)
def needsJSTypes(self):
for p
in self.productions:
if p.kind ==
"interface" and p.needsJSTypes():
return True
return False
class CDATA(object):
kind =
"cdata"
_re = re.compile(r
"\n+")
def __init__(self, data, location):
self.data = self._re.sub(
"\n", data)
self.location = location
def resolve(self, parent):
# This can be a false-positive if the word `virtual` is included in a
# comment, however this doesn't seem to happen very often.
if isinstance(parent, Interface)
and re.search(r
"\bvirtual\b", self.data):
raise IDLError(
"cannot declare a C++ `virtual` member in XPIDL interface",
self.location,
notes=textwrap.fill(
"""All virtual members must be declared directly using XPIDL.
Both the Rust bindings
and XPConnect rely on the per-platform
vtable layouts generated by the XPIDL compiler to allow
cross-language XPCOM method calls between JS
and C++.
Consider using a `[notxpcom, nostdcall]` method instead.
"""
),
)
def __str__(self):
return "cdata: %s\n\t%r\n" % (self.location.get(), self.data)
def count(self):
return 0
class Typedef(object):
kind =
"typedef"
def __init__(self, type, name, attlist, location, doccomments):
self.type = type
self.name = name
self.location = location
self.doccomments = doccomments
def __eq__(self, other):
return self.name == other.name
and self.type == other.type
def resolve(self, parent):
parent.setName(self)
self.realtype = parent.getName(self.type, self.location)
if not isinstance(self.realtype, (Builtin, CEnum, Native, Typedef)):
raise IDLError(
"Unsupported typedef target type", self.location)
def nativeType(self, calltype):
return "%s %s" % (self.name,
"*" if "out" in calltype
else "")
def rustType(self, calltype):
return "%s%s" % (
"*mut " if "out" in calltype
else "", self.name)
def tsType(self):
# Make sure that underlying type is supported: doesn't throw TSNoncompat.
self.realtype.tsType()
return self.name
def __str__(self):
return "typedef %s %s\n" % (self.type, self.name)
class Forward(object):
kind =
"forward"
def __init__(self, name, location, doccomments):
self.name = name
self.location = location
self.doccomments = doccomments
def __eq__(self, other):
return other.kind ==
"forward" and other.name == self.name
def resolve(self, parent):
# Hack alert: if an identifier is already present, move the doccomments
# forward.
if parent.hasName(self.name):
for i
in range(0, len(parent.productions)):
if parent.productions[i]
is self:
break
for i
in range(i + 1, len(parent.productions)):
if hasattr(parent.productions[i],
"doccomments"):
parent.productions[i].doccomments[0:0] = self.doccomments
break
parent.setName(self)
def nativeType(self, calltype):
if calltype ==
"element":
return "RefPtr<%s>" % self.name
return "%s *%s" % (self.name,
"*" if "out" in calltype
else "")
def rustType(self, calltype):
if rustPreventForward(self.name):
raise RustNoncompat(
"forward declaration %s is unsupported" % self.name)
if calltype ==
"element":
return "Option>" % self.name
return "%s*const %s" % (
"*mut" if "out" in calltype
else "", self.name)
def tsType(self):
return self.name
def __str__(self):
return "forward-declared %s\n" % self.name
class Native(object):
kind =
"native"
modifier =
None
specialtype =
None
# A tuple type here means that a custom value is used for each calltype:
# (in, out/inout, array element) respectively.
# A `None` here means that the written type should be used as-is.
specialtypes = {
"nsid":
None,
"utf8string": (
"const nsACString&",
"nsACString&",
"nsCString"),
"cstring": (
"const nsACString&",
"nsACString&",
"nsCString"),
"astring": (
"const nsAString&",
"nsAString&",
"nsString"),
"jsval": (
"JS::Handle",
"JS::MutableHandle",
"JS::Value"),
"promise":
"::mozilla::dom::Promise",
}
def __init__(self, name, nativename, attlist, location):
self.name = name
self.nativename = nativename
self.location = location
for name, value, aloc
in attlist:
if value
is not None:
raise IDLError(
"Unexpected attribute value", aloc)
if name
in (
"ptr",
"ref"):
if self.modifier
is not None:
raise IDLError(
"More than one ptr/ref modifier", aloc)
self.modifier = name
elif name
in self.specialtypes.keys():
if self.specialtype
is not None:
raise IDLError(
"More than one special type", aloc)
self.specialtype = name
if self.specialtypes[name]
is not None:
self.nativename = self.specialtypes[name]
else:
raise IDLError(
"Unexpected attribute", aloc)
def __eq__(self, other):
return (
self.name == other.name
and self.nativename == other.nativename
and self.modifier == other.modifier
and self.specialtype == other.specialtype
)
def resolve(self, parent):
parent.setName(self)
def isPtr(self, calltype):
return self.modifier ==
"ptr"
def isRef(self, calltype):
return self.modifier ==
"ref"
def nativeType(self, calltype, const=
False, shared=
False):
if shared:
if calltype !=
"out":
raise IDLError(
"[shared] only applies to out parameters.", self.location
)
const =
True
if isinstance(self.nativename, tuple):
if calltype ==
"in":
return self.nativename[0] +
" "
elif "out" in calltype:
return self.nativename[1] +
" "
else:
return self.nativename[2] +
" "
# 'in' nsid parameters should be made 'const'
if self.specialtype ==
"nsid" and calltype ==
"in":
const =
True
if calltype ==
"element":
if self.specialtype ==
"nsid":
if self.isPtr(calltype):
raise IDLError(
"Array not yet supported. "
"File an XPConnect bug if you need it.",
self.location,
)
# ns[CI]?IDs should be held directly in Array<T>s
return self.nativename
if self.isRef(calltype):
raise IDLError(
"[ref] qualified type unsupported in Array", self.location
)
# Promises should be held in RefPtr<T> in Array<T>s
if self.specialtype ==
"promise":
return "RefPtr"
if self.isRef(calltype):
m =
"& " # [ref] is always passed with a single indirection
else:
m =
"* " if "out" in calltype
else ""
if self.isPtr(calltype):
m +=
"* "
return "%s%s %s" % (const
and "const " or "", self.nativename, m)
def rustType(self, calltype, const=
False, shared=
False):
# For the most part, 'native' types don't make sense in rust, as they
# are native C++ types. However, we can support a few types here, as
# they're important and can easily be translated.
#
# NOTE: This code doesn't try to perfectly match C++ constness, as
# constness doesn't affect ABI, and raw pointers are already unsafe.
if self.modifier
not in [
"ptr",
"ref"]:
raise RustNoncompat(
"Rust only supports [ref] / [ptr] native types")
if shared:
if calltype !=
"out":
raise IDLError(
"[shared] only applies to out parameters.", self.location
)
const =
True
# 'in' nsid parameters should be made 'const'
if self.specialtype ==
"nsid" and calltype ==
"in":
const =
True
prefix =
"*const " if const
or shared
else "*mut "
if "out" in calltype
and self.isPtr(calltype):
prefix =
"*mut " + prefix
if self.specialtype:
# The string types are very special, and need to be handled seperately.
if self.specialtype
in [
"cstring",
"utf8string"]:
if calltype ==
"in":
return "*const ::nsstring::nsACString"
elif "out" in calltype:
return "*mut ::nsstring::nsACString"
else:
return "::nsstring::nsCString"
if self.specialtype ==
"astring":
if calltype ==
"in":
return "*const ::nsstring::nsAString"
elif "out" in calltype:
return "*mut ::nsstring::nsAString"
else:
return "::nsstring::nsString"
# nsid has some special handling, but generally re-uses the generic
# prefix handling above.
if self.specialtype ==
"nsid":
if "element" in calltype:
if self.isPtr(calltype):
raise IDLError(
"Array not yet supported. "
"File an XPConnect bug if you need it.",
self.location,
)
return self.nativename
return prefix + self.nativename
raise RustNoncompat(
"special type %s unsupported" % self.specialtype)
# These 3 special types correspond to native pointer types which can
# generally be supported behind pointers. Other types are not supported
# for now.
if self.nativename ==
"void":
return prefix +
"libc::c_void"
if self.nativename ==
"char":
return prefix +
"libc::c_char"
if self.nativename ==
"char16_t":
return prefix +
"u16"
raise RustNoncompat(
"native type %s unsupported" % self.nativename)
ts_special = {
"astring":
"string",
"cstring":
"string",
"jsval":
"any",
"nsid":
"nsID",
"promise":
"Promise",
"utf8string":
"string",
}
def tsType(self):
if type := self.ts_special.get(self.specialtype,
None):
return type
raise TSNoncompat(f
"Native type {self.name} unsupported in TypeScript")
def __str__(self):
return "native %s(%s)\n" % (self.name, self.nativename)
class WebIDL(object):
kind =
"webidl"
def __init__(self, name, location):
self.name = name
self.location = location
def __eq__(self, other):
return other.kind ==
"webidl" and self.name == other.name
def resolve(self, parent):
# XXX(nika): We don't handle _every_ kind of webidl object here (as that
# would be hard). For example, we don't support nsIDOM*-defaulting
# interfaces.
# TODO: More explicit compile-time checks?
assert (
parent.webidlconfig
is not None
),
"WebIDL declarations require passing webidlconfig to resolve."
# Resolve our native name according to the WebIDL configs.
config = parent.webidlconfig.get(self.name, {})
self.native = config.get(
"nativeType")
if self.native
is None:
self.native =
"mozilla::dom::%s" % self.name
self.headerFile = config.get(
"headerFile")
if self.headerFile
is None:
self.headerFile = self.native.replace(
"::",
"/") +
".h"
parent.setName(self)
def nativeType(self, calltype, const=
False):
if calltype ==
"element":
return "RefPtr<%s%s>" % (
"const " if const
else "", self.native)
return "%s%s *%s" % (
"const " if const
else "",
self.native,
"*" if "out" in calltype
else "",
)
def rustType(self, calltype, const=
False):
# Just expose the type as a void* - we can't do any better.
return "%s*const libc::c_void" % (
"*mut " if "out" in calltype
else "")
def tsType(self):
return self.name
def __str__(self):
return "webidl %s\n" % self.name
class Interface(object):
kind =
"interface"
def __init__(self, name, attlist, base, members, location, doccomments):
self.name = name
self.attributes = InterfaceAttributes(attlist, location)
self.base = base
self.members = members
self.location = location
self.namemap = NameMap()
self.doccomments = doccomments
self.nativename = name
for m
in members:
if not isinstance(m, CDATA):
self.namemap.set(m)
def __eq__(self, other):
return self.name == other.name
and self.location == other.location
def resolve(self, parent):
self.idl = parent
if not self.attributes.scriptable
and self.attributes.builtinclass:
raise IDLError(
"Non-scriptable interface '%s' doesn't need to be marked builtinclass"
% self.name,
self.location,
)
# Hack alert: if an identifier is already present, libIDL assigns
# doc comments incorrectly. This is quirks-mode extraordinaire!
if parent.hasName(self.name):
for member
in self.members:
if hasattr(member,
"doccomments"):
member.doccomments[0:0] = self.doccomments
break
self.doccomments = parent.getName(TypeId(self.name),
None).doccomments
if self.attributes.function:
has_method =
False
for member
in self.members:
if member.kind ==
"method":
if has_method:
raise IDLError(
"interface '%s' has multiple methods, but marked 'function'"
% self.name,
self.location,
)
else:
has_method =
True
parent.setName(self)
if self.base
is not None:
realbase = parent.getName(TypeId(self.base), self.location)
if realbase.kind !=
"interface":
raise IDLError(
"interface '%s' inherits from non-interface type '%s'"
% (self.name, self.base),
self.location,
)
if self.attributes.scriptable
and not realbase.attributes.scriptable:
raise IDLError(
"interface '%s' is scriptable but derives from "
"non-scriptable '%s'" % (self.name, self.base),
self.location,
warning=
True,
)
if (
self.attributes.scriptable
and realbase.attributes.builtinclass
and not self.attributes.builtinclass
):
raise IDLError(
"interface '%s' is not builtinclass but derives from "
"builtinclass '%s'" % (self.name, self.base),
self.location,
)
if realbase.attributes.rust_sync
and not self.attributes.rust_sync:
raise IDLError(
"interface '%s' is not rust_sync but derives from rust_sync '%s'"
% (self.name, self.base),
self.location,
)
if (
self.attributes.rust_sync
and self.attributes.scriptable
and not self.attributes.builtinclass
):
raise IDLError(
"interface '%s' is rust_sync but is not builtinclass" % self.name,
self.location,
)
elif self.name !=
"nsISupports":
raise IDLError(
"Interface '%s' must inherit from nsISupports" % self.name,
self.location,
)
for member
in self.members:
member.resolve(self)
# The number 250 is NOT arbitrary; this number is the maximum number of
# stub entries defined in xpcom/reflect/xptcall/genstubs.pl
# Do not increase this value without increasing the number in that
# location, or you WILL cause otherwise unknown problems!
if self.countEntries() > 250
and not self.attributes.builtinclass:
raise IDLError(
"interface '%s' has too many entries" % self.name, self.location
)
def nativeType(self, calltype, const=
False):
if calltype ==
"element":
return "RefPtr<%s>" % self.name
return "%s%s *%s" % (
"const " if const
else "",
self.name,
"*" if "out" in calltype
else "",
)
def rustType(self, calltype, const=
False):
if calltype ==
"element":
return "Option>" % self.name
return "%s*const %s" % (
"*mut " if "out" in calltype
else "", self.name)
def __str__(self):
l = [
"interface %s\n" % self.name]
if self.base
is not None:
l.append(
"\tbase %s\n" % self.base)
l.append(str(self.attributes))
if self.members
is None:
l.append(
"\tincomplete type\n")
else:
for m
in self.members:
l.append(str(m))
return "".join(l)
def getConst(self, name, location):
# The constant may be in a base class
iface = self
while name
not in iface.namemap
and iface.base
is not None:
iface = self.idl.getName(TypeId(iface.base), self.location)
if name
not in iface.namemap:
raise IDLError(
"cannot find symbol '%s'" % name, location)
c = iface.namemap.get(name, location)
if c.kind !=
"const":
raise IDLError(
"symbol '%s' is not a constant" % name, location)
return c.getValue()
def needsJSTypes(self):
for m
in self.members:
if m.kind ==
"attribute" and m.type == TypeId(
"jsval"):
return True
if m.kind ==
"method" and m.needsJSTypes():
return True
return False
def countEntries(self):
"""Returns the number of entries in the vtable for this interface."""
total = sum(member.count()
for member
in self.members)
if self.base
is not None:
realbase = self.idl.getName(TypeId(self.base), self.location)
total += realbase.countEntries()
return total
def tsType(self):
return self.name
class InterfaceAttributes(object):
uuid =
None
scriptable =
False
builtinclass =
False
function =
False
main_process_scriptable_only =
False
rust_sync =
False
def setuuid(self, value):
self.uuid = value.lower()
def setscriptable(self):
self.scriptable =
True
def setfunction(self):
self.function =
True
def setbuiltinclass(self):
self.builtinclass =
True
def setmain_process_scriptable_only(self):
self.main_process_scriptable_only =
True
def setrust_sync(self):
self.rust_sync =
True
actions = {
"uuid": (
True, setuuid),
"scriptable": (
False, setscriptable),
"builtinclass": (
False, setbuiltinclass),
"function": (
False, setfunction),
"object": (
False,
lambda self:
True),
"main_process_scriptable_only": (
False, setmain_process_scriptable_only),
"rust_sync": (
False, setrust_sync),
}
def __init__(self, attlist, location):
def badattribute(self):
raise IDLError(
"Unexpected interface attribute '%s'" % name, location)
for name, val, aloc
in attlist:
hasval, action = self.actions.get(name, (
False, badattribute))
if hasval:
if val
is None:
raise IDLError(
"Expected value for attribute '%s'" % name, aloc)
action(self, val)
else:
if val
is not None:
raise IDLError(
"Unexpected value for attribute '%s'" % name, aloc)
action(self)
if self.uuid
is None:
raise IDLError(
"interface has no uuid", location)
def __str__(self):
l = []
if self.uuid:
l.append(
"\tuuid: %s\n" % self.uuid)
if self.scriptable:
l.append(
"\tscriptable\n")
if self.builtinclass:
l.append(
"\tbuiltinclass\n")
if self.function:
l.append(
"\tfunction\n")
if self.main_process_scriptable_only:
l.append(
"\tmain_process_scriptable_only\n")
if self.rust_sync:
l.append(
"\trust_sync\n")
return "".join(l)
class ConstMember(object):
kind =
"const"
def __init__(self, type, name, value, location, doccomments):
self.type = type
self.name = name
self.valueFn = value
self.location = location
self.doccomments = doccomments
def resolve(self, parent):
self.realtype = parent.idl.getName(self.type, self.location)
self.iface = parent
basetype = self.realtype
while isinstance(basetype, Typedef):
basetype = basetype.realtype
if not isinstance(basetype, Builtin)
or not basetype.maybeConst:
raise IDLError(
"const may only be an integer type, not %s" % self.type.name,
self.location,
)
self.basetype = basetype
# Value is a lambda. Resolve it.
self.value = self.valueFn(self.iface)
min_val = -(2**31)
if basetype.signed
else 0
max_val = 2**31 - 1
if basetype.signed
else 2**32 - 1
if self.value < min_val
or self.value > max_val:
raise IDLError(
"xpidl constants must fit within %s"
% (
"int32_t" if basetype.signed
else "uint32_t"),
self.location,
)
def getValue(self):
return self.value
def __str__(self):
return "\tconst %s %s = %s\n" % (self.type, self.name, self.getValue())
def count(self):
return 0
# Represents a single name/value pair in a CEnum
class CEnumVariant(object):
# Treat CEnumVariants as consts in terms of value resolution, so we can
# do things like binary operation values for enum members.
kind =
"const"
def __init__(self, name, value, location):
self.name = name
self.valueFn = value
self.location = location
def getValue(self):
return self.value
class CEnum(object):
kind =
"cenum"
def __init__(self, width, name, variants, location, doccomments):
# We have to set a name here, otherwise we won't pass namemap checks on
# the interface. This name will change it in resolve(), in order to
# namespace the enum within the interface.
self.name = name
self.basename = name
self.width = width
self.location = location
self.namemap = NameMap()
self.doccomments = doccomments
self.variants = variants
if self.width
not in (8, 16, 32):
raise IDLError(
"Width must be one of {8, 16, 32}", self.location)
def resolve(self, iface):
self.iface = iface
# Renaming enum to faux-namespace the enum type to the interface in JS
# so we don't collide in the global namespace. Hacky/ugly but it does
# the job well enough, and the name will still be interface::variant in
# C++.
self.name =
"%s_%s" % (self.iface.name, self.basename)
self.iface.idl.setName(self)
# Compute the value for each enum variant that doesn't set its own
# value
next_value = 0
for variant
in self.variants:
# CEnum variants resolve to interface level consts in javascript,
# meaning their names could collide with other interface members.
# Iterate through all CEnum variants to make sure there are no
# collisions.
self.iface.namemap.set(variant)
# Value may be a lambda. If it is, resolve it.
if variant.valueFn:
next_value = variant.value = variant.valueFn(self.iface)
else:
variant.value = next_value
next_value += 1
def count(self):
return 0
def nativeType(self, calltype):
if "out" in calltype:
return "%s::%s *" % (self.iface.name, self.basename)
return "%s::%s " % (self.iface.name, self.basename)
def rustType(self, calltype):
return "%s u%d" % (
"*mut" if "out" in calltype
else "", self.width)
def tsType(self):
return f
"{self.iface.name}.{self.basename}"
def __str__(self):
body =
", ".join(
"%s = %s" % v
for v
in self.variants)
return "\tcenum %s : %d { %s };\n" % (self.name, self.width, body)
# Infallible doesn't work for all return types.
#
# It also must be implemented on a builtinclass (otherwise it'd be unsound as
# it could be implemented by JS).
def ensureInfallibleIsSound(methodOrAttribute):
if not methodOrAttribute.infallible:
return
if methodOrAttribute.realtype.kind
not in [
"builtin",
"interface",
"forward",
"webidl",
"cenum",
]:
raise IDLError(
"[infallible] only works on interfaces, domobjects, and builtin types "
"(numbers, booleans, cenum, and raw char types)",
methodOrAttribute.location,
)
ifaceAttributes = methodOrAttribute.iface.attributes
if ifaceAttributes.scriptable
and not ifaceAttributes.builtinclass:
raise IDLError(
"[infallible] attributes and methods are only allowed on "
"non-[scriptable] or [builtinclass] interfaces",
methodOrAttribute.location,
)
if methodOrAttribute.notxpcom:
raise IDLError(
"[infallible] does not make sense for a [notxpcom] method or attribute",
methodOrAttribute.location,
)
# An interface cannot be implemented by JS if it has a notxpcom or nostdcall
# method or attribute, or uses a by-value native type, so it must be marked as
# builtinclass.
def ensureBuiltinClassIfNeeded(methodOrAttribute):
iface = methodOrAttribute.iface
if not iface.attributes.scriptable
or iface.attributes.builtinclass:
return
if iface.name ==
"nsISupports":
return
# notxpcom and nostdcall types change calling conventions, which breaks
# xptcall wrappers. We cannot allow XPCWrappedJS to be created for
# interfaces with these methods.
if methodOrAttribute.notxpcom:
raise IDLError(
(
"scriptable interface '%s' must be marked [builtinclass] because it "
"contains a [notxpcom] %s '%s'"
)
% (iface.name, methodOrAttribute.kind, methodOrAttribute.name),
methodOrAttribute.location,
)
if methodOrAttribute.nostdcall:
raise IDLError(
(
"scriptable interface '%s' must be marked [builtinclass] because it "
"contains a [nostdcall] %s '%s'"
)
% (iface.name, methodOrAttribute.kind, methodOrAttribute.name),
methodOrAttribute.location,
)
# Methods with custom native parameters passed without indirection cannot be
# safely handled by xptcall (as it cannot know the calling stack/register
# layout), so require the interface to be builtinclass.
#
# Only "in" parameters and writable attributes are checked, as other
# parameters are always passed indirectly, so do not impact calling
# conventions.
def typeNeedsBuiltinclass(type):
inner = type
while inner.kind ==
"typedef":
inner = inner.realtype
return (
inner.kind ==
"native"
and inner.specialtype
is None
and inner.modifier
is None
)
if methodOrAttribute.kind ==
"method":
for p
in methodOrAttribute.params:
if p.paramtype ==
"in" and typeNeedsBuiltinclass(p.realtype):
raise IDLError(
(
"scriptable interface '%s' must be marked [builtinclass] "
"because it contains method '%s' with a by-value custom native "
"parameter '%s'"
)
% (iface.name, methodOrAttribute.name, p.name),
methodOrAttribute.location,
)
elif methodOrAttribute.kind ==
"attribute" and not methodOrAttribute.readonly:
if typeNeedsBuiltinclass(methodOrAttribute.realtype):
raise IDLError(
(
"scriptable interface '%s' must be marked [builtinclass] because it "
"contains writable attribute '%s' with a by-value custom native type"
)
% (iface.name, methodOrAttribute.name),
methodOrAttribute.location,
)
def ensureNoscriptIfNeeded(methodOrAttribute):
if not methodOrAttribute.isScriptable():
return
# NOTE: We can't check forward-declared interfaces to see if they're
# scriptable, as the information about whether they're scriptable is not
# known here.
def typeNeedsNoscript(type):
if type.kind
in [
"array",
"legacyarray"]:
return typeNeedsNoscript(type.type)
if type.kind ==
"typedef":
return typeNeedsNoscript(type.realtype)
if type.kind ==
"native":
return type.specialtype
is None
if type.kind ==
"interface":
return not type.attributes.scriptable
return False
if typeNeedsNoscript(methodOrAttribute.realtype):
raise IDLError(
"%s '%s' must be marked [noscript] because it has a non-scriptable type"
% (methodOrAttribute.kind, methodOrAttribute.name),
methodOrAttribute.location,
)
if methodOrAttribute.kind ==
"method":
for p
in methodOrAttribute.params:
# iid_is arguments have their type ignored, so shouldn't be checked.
if not p.iid_is
and typeNeedsNoscript(p.realtype):
raise IDLError(
(
"method '%s' must be marked [noscript] because it has a "
"non-scriptable parameter '%s'"
)
% (methodOrAttribute.name, p.name),
methodOrAttribute.location,
)
class Attribute(object):
kind =
"attribute"
noscript =
False
notxpcom =
False
readonly =
False
symbol =
False
implicit_jscontext =
False
nostdcall =
False
must_use =
False
binaryname =
None
infallible =
False
# explicit_setter_can_run_script is true if the attribute is explicitly
# annotated as having a setter that can cause script to run.
explicit_setter_can_run_script =
False
# explicit_getter_can_run_script is true if the attribute is explicitly
# annotated as having a getter that can cause script to run.
explicit_getter_can_run_script =
False
def __init__(self, type, name, attlist, readonly, location, doccomments):
self.type = type
self.name = name
self.attlist = attlist
self.readonly = readonly
self.location = location
self.doccomments = doccomments
for name, value, aloc
in attlist:
if name ==
"binaryname":
if value
is None:
raise IDLError(
"binaryname attribute requires a value", aloc)
self.binaryname = value
continue
if value
is not None:
raise IDLError(
"Unexpected attribute value", aloc)
if name ==
"noscript":
self.noscript =
True
elif name ==
"notxpcom":
self.notxpcom =
True
elif name ==
"symbol":
self.symbol =
True
elif name ==
"implicit_jscontext":
self.implicit_jscontext =
True
elif name ==
"nostdcall":
self.nostdcall =
True
elif name ==
"must_use":
self.must_use =
True
elif name ==
"infallible":
self.infallible =
True
elif name ==
"can_run_script":
if (
self.explicit_setter_can_run_script
or self.explicit_getter_can_run_script
):
raise IDLError(
"Redundant getter_can_run_script or "
"setter_can_run_script annotation on "
"attribute",
aloc,
)
self.explicit_setter_can_run_script =
True
self.explicit_getter_can_run_script =
True
elif name ==
"setter_can_run_script":
if self.explicit_setter_can_run_script:
raise IDLError(
"Redundant setter_can_run_script annotation " "on attribute",
aloc,
)
self.explicit_setter_can_run_script =
True
elif name ==
"getter_can_run_script":
if self.explicit_getter_can_run_script:
raise IDLError(
"Redundant getter_can_run_script annotation " "on attribute",
aloc,
)
self.explicit_getter_can_run_script =
True
else:
raise IDLError(
"Unexpected attribute '%s'" % name, aloc)
def resolve(self, iface):
self.iface = iface
self.realtype = iface.idl.getName(self.type, self.location)
ensureInfallibleIsSound(self)
ensureBuiltinClassIfNeeded(self)
ensureNoscriptIfNeeded(self)
def toIDL(self):
attribs = attlistToIDL(self.attlist)
readonly = self.readonly
and "readonly " or ""
return "%s%sattribute %s %s;" % (attribs, readonly, self.type, self.name)
def isScriptable(self):
if not self.iface.attributes.scriptable:
return False
return not (self.noscript
or self.notxpcom
or self.nostdcall)
def __str__(self):
return "\t%sattribute %s %s\n" % (
self.readonly
and "readonly " or "",
self.type,
self.name,
)
def count(self):
return self.readonly
and 1
or 2
class Method(object):
kind =
"method"
noscript =
False
notxpcom =
False
symbol =
False
binaryname =
None
implicit_jscontext =
False
nostdcall =
False
must_use =
False
optional_argc =
False
# explicit_can_run_script is true if the method is explicitly annotated
# as being able to cause script to run.
explicit_can_run_script =
False
infallible =
False
def __init__(self, type, name, attlist, paramlist, location, doccomments, raises):
self.type = type
self.name = name
self.attlist = attlist
self.params = paramlist
self.location = location
self.doccomments = doccomments
self.raises = raises
for name, value, aloc
in attlist:
if name ==
"binaryname":
if value
is None:
raise IDLError(
"binaryname attribute requires a value", aloc)
self.binaryname = value
continue
if value
is not None:
raise IDLError(
"Unexpected attribute value", aloc)
if name ==
"noscript":
self.noscript =
True
elif name ==
"notxpcom":
self.notxpcom =
True
elif name ==
"symbol":
self.symbol =
True
elif name ==
"implicit_jscontext":
self.implicit_jscontext =
True
elif name ==
"optional_argc":
self.optional_argc =
True
elif name ==
"nostdcall":
self.nostdcall =
True
elif name ==
"must_use":
self.must_use =
True
elif name ==
"can_run_script":
self.explicit_can_run_script =
True
elif name ==
"infallible":
self.infallible =
True
else:
raise IDLError(
"Unexpected attribute '%s'" % name, aloc)
self.namemap = NameMap()
for p
in paramlist:
self.namemap.set(p)
def resolve(self, iface):
self.iface = iface
self.realtype = self.iface.idl.getName(self.type, self.location)
for p
in self.params:
p.resolve(self)
for p
in self.params:
if p.retval
and p != self.params[-1]:
raise IDLError(
"'retval' parameter '%s' is not the last parameter" % p.name,
self.location,
)
if p.size_is:
size_param = self.namemap.get(p.size_is, p.location)
if (
p.paramtype.count(
"in") == 1
and size_param.paramtype.count(
"in") == 0
):
raise IDLError(
"size_is parameter of an input must also be an input",
p.location,
)
if getBuiltinOrNativeTypeName(size_param.realtype) !=
"uint32_t":
raise IDLError(
"size_is parameter must have type 'uint32_t'",
p.location,
)
if p.iid_is:
iid_param = self.namemap.get(p.iid_is, p.location)
if (
p.paramtype.count(
"in") == 1
and iid_param.paramtype.count(
"in") == 0
):
raise IDLError(
"iid_is parameter of an input must also be an input",
p.location,
)
if getBuiltinOrNativeTypeName(iid_param.realtype) !=
"[nsid]":
raise IDLError(
"iid_is parameter must be an nsIID",
self.location,
)
ensureInfallibleIsSound(self)
ensureBuiltinClassIfNeeded(self)
ensureNoscriptIfNeeded(self)
def isScriptable(self):
if not self.iface.attributes.scriptable:
return False
return not (self.noscript
or self.notxpcom
or self.nostdcall)
def __str__(self):
return "\t%s %s(%s)\n" % (
self.type,
self.name,
", ".join([p.name
for p
in self.params]),
)
def toIDL(self):
if len(self.raises):
raises =
" raises (%s)" %
",".join(self.raises)
else:
raises =
""
return "%s%s %s (%s)%s;" % (
attlistToIDL(self.attlist),
self.type,
self.name,
", ".join([p.toIDL()
for p
in self.params]),
raises,
)
def needsJSTypes(self):
if self.implicit_jscontext:
return True
if self.type == TypeId(
"jsval"):
return True
for p
in self.params:
t = p.realtype
if isinstance(t, Native)
and t.specialtype ==
"jsval":
return True
return False
def count(self):
return 1
class Param(object):
size_is =
None
iid_is =
None
const =
False
array =
False
retval =
False
shared =
False
optional =
False
default_value =
None
def __init__(self, paramtype, type, name, attlist, location, realtype=
None):
self.paramtype = paramtype
self.type = type
self.name = name
self.attlist = attlist
self.location = location
self.realtype = realtype
for name, value, aloc
in attlist:
# Put the value-taking attributes first!
if name ==
"size_is":
if value
is None:
raise IDLError(
"'size_is' must specify a parameter", aloc)
self.size_is = value
elif name ==
"iid_is":
if value
is None:
raise IDLError(
"'iid_is' must specify a parameter", aloc)
self.iid_is = value
elif name ==
"default":
if value
is None:
raise IDLError(
"'default' must specify a default value", aloc)
self.default_value = value
else:
if value
is not None:
raise IDLError(
"Unexpected value for attribute '%s'" % name, aloc)
if name ==
"const":
self.const =
True
elif name ==
"array":
self.array =
True
elif name ==
"retval":
self.retval =
True
elif name ==
"shared":
self.shared =
True
elif name ==
"optional":
self.optional =
True
else:
raise IDLError(
"Unexpected attribute '%s'" % name, aloc)
def resolve(self, method):
self.realtype = method.iface.idl.getName(self.type, self.location)
if self.array:
self.realtype = LegacyArray(self.realtype)
def nativeType(self):
kwargs = {}
if self.shared:
kwargs[
"shared"] =
True
if self.const:
kwargs[
"const"] =
True
try:
return self.realtype.nativeType(self.paramtype, **kwargs)
except IDLError
as e:
raise IDLError(str(e), self.location)
except TypeError:
raise IDLError(
"Unexpected parameter attribute", self.location)
def rustType(self):
kwargs = {}
if self.shared:
kwargs[
"shared"] =
True
if self.const:
kwargs[
"const"] =
True
try:
return self.realtype.rustType(self.paramtype, **kwargs)
except IDLError
as e:
raise IDLError(str(e), self.location)
except TypeError:
raise IDLError(
"Unexpected parameter attribute", self.location)
def toIDL(self):
return "%s%s %s %s" % (
paramAttlistToIDL(self.attlist),
self.paramtype,
self.type,
self.name,
)
def tsType(self):
# A generic retval param type needs special handling.
if self.retval
and self.iid_is:
return "nsQIResult"
type = self.realtype.tsType()
if self.paramtype ==
"inout":
return f
"InOutParam<{type}>"
if self.paramtype ==
"out":
return f
"OutParam<{type}>"
return type
class LegacyArray(object):
kind =
"legacyarray"
def __init__(self, basetype):
self.type = basetype
self.location = self.type.location
def nativeType(self, calltype, const=
False):
if "element" in calltype:
raise IDLError(
"nested [array] unsupported", self.location)
# For legacy reasons, we have to add a 'const ' to builtin pointer array
# types. (`[array] in string` and `[array] in wstring` parameters)
if (
calltype ==
"in"
and isinstance(self.type, Builtin)
and self.type.isPointer()
):
const =
True
return "%s%s*%s" % (
"const " if const
else "",
self.type.nativeType(
"legacyelement"),
"*" if "out" in calltype
else "",
)
def rustType(self, calltype, const=
False):
return "%s%s%s" % (
"*mut " if "out" in calltype
else "",
"*const " if const
else "*mut ",
self.type.rustType(
"legacyelement"),
)
def tsType(self):
return self.type.tsType() +
"[]"
class Array(object):
kind =
"array"
def __init__(self, type, location):
self.type = type
self.location = location
@property
def name(self):
return "Array<%s>" % self.type.name
def resolve(self, idl):
idl.getName(self.type, self.location)
def nativeType(self, calltype):
if calltype ==
"legacyelement":
raise IDLError(
"[array] Array is unsupported", self.location)
base =
"nsTArray<%s>" % self.type.nativeType(
"element")
if "out" in calltype:
return "%s& " % base
elif "in" == calltype:
return "const %s& " % base
else:
return base
def rustType(self, calltype):
if calltype ==
"legacyelement":
raise IDLError(
"[array] Array is unsupported", self.location)
base =
"thin_vec::ThinVec<%s>" % self.type.rustType(
"element")
if "out" in calltype:
return "*mut %s" % base
elif "in" == calltype:
return "*const %s" % base
else:
return base
def tsType(self):
return self.type.tsType() +
"[]"
TypeId = namedtuple(
"TypeId",
"name params")
# Make str(TypeId) produce a nicer value
TypeId.__str__ =
lambda self: (
"%s<%s>" % (self.name,
", ".join(str(p)
for p
in self.params))
if self.params
is not None
else self.name
)
# Allow skipping 'params' in TypeId(..)
TypeId.__new__.__defaults__ = (
None,)
class IDLParser(object):
keywords = {
"cenum":
"CENUM",
"const":
"CONST",
"interface":
"INTERFACE",
"in":
"IN",
"inout":
"INOUT",
"out":
"OUT",
"attribute":
"ATTRIBUTE",
"raises":
"RAISES",
"readonly":
"READONLY",
"native":
"NATIVE",
"typedef":
"TYPEDEF",
"webidl":
"WEBIDL",
}
tokens = [
"IDENTIFIER",
"CDATA",
"INCLUDE",
"IID",
"NUMBER",
"HEXNUM",
"LSHIFT",
"RSHIFT",
"NATIVEID",
]
tokens.extend(keywords.values())
states = ((
"nativeid",
"exclusive"),)
hexchar = r
"[a-fA-F0-9]"
t_NUMBER = r
"-?\d+"
t_HEXNUM = r
"0x%s+" % hexchar
t_LSHIFT = r
"<<"
t_RSHIFT = r
">>"
literals =
'"(){}[]<>,;:=|+-*'
t_ignore =
" \t"
def t_multilinecomment(self, t):
r
"/\*(\n|.)*?\*/"
t.lexer.lineno += t.value.count(
"\n")
if t.value.startswith(
"/**"):
self._doccomments.append(t.value)
def t_singlelinecomment(self, t):
r
"//[^\n]*"
def t_IID(self, t):
return t
t_IID.__doc__ = r
"%(c)s{8}-%(c)s{4}-%(c)s{4}-%(c)s{4}-%(c)s{12}" % {
"c": hexchar}
def t_IDENTIFIER(self, t):
r
"(unsigned\ long\ long|unsigned\ short|unsigned\ long|long\ long)(?!_?[A-Za-z][A-Za-z_0-9])|_?[A-Za-z][A-Za-z_0-9]*" # NOQA: E501
t.type = self.keywords.get(t.value,
"IDENTIFIER")
return t
def t_LCDATA(self, t):
r
"%\{[ ]*C\+\+[ ]*\n(?P(\n|.)*?\n?)%\}[ ]*(C\+\+)?"
t.type =
"CDATA"
t.value = t.lexer.lexmatch.group(
"cdata")
t.lexer.lineno += t.value.count(
"\n")
return t
def t_INCLUDE(self, t):
r
'\#include[ \t]+"[^"\n]+"'
inc, value, end = t.value.split(
'"')
t.value = value
return t
def t_directive(self, t):
r
"\#(?P[a-zA-Z]+)[^\n]+"
raise IDLError(
"Unrecognized directive %s" % t.lexer.lexmatch.group(
"directive"),
Location(
lexer=self.lexer, lineno=self.lexer.lineno, lexpos=self.lexer.lexpos
),
)
def t_newline(self, t):
r
"\n+"
t.lexer.lineno += len(t.value)
def t_nativeid_NATIVEID(self, t):
# Matches non-parenthesis characters, or a single open and closing
# parenthesis with at least one non-parenthesis character before,
# between and after them (for compatibility with std::function).
r
"[^()\n]+(?:\([^()\n]+\)[^()\n]+)?(?=\))"
t.lexer.begin(
"INITIAL")
return t
t_nativeid_ignore =
""
def t_ANY_error(self, t):
raise IDLError(
"unrecognized input",
Location(
lexer=self.lexer, lineno=self.lexer.lineno, lexpos=self.lexer.lexpos
),
)
precedence = (
(
"left",
"|"),
(
"left",
"LSHIFT",
"RSHIFT"),
(
"left",
"+",
"-"),
(
"left",
"*"),
(
"left",
"UMINUS"),
)
def p_idlfile(self, p):
"""idlfile : productions"""
p[0] = IDL(p[1])
def p_productions_start(self, p):
"""productions :"""
p[0] = []
def p_productions_cdata(self, p):
"""productions : CDATA productions"""
p[0] = list(p[2])
p[0].insert(0, CDATA(p[1], self.getLocation(p, 1)))
def p_productions_include(self, p):
"""productions : INCLUDE productions"""
p[0] = list(p[2])
p[0].insert(0, Include(p[1], self.getLocation(p, 1)))
def p_productions_interface(self, p):
"""productions : interface productions
| typedef productions
| native productions
| webidl productions
"""
p[0] = list(p[2])
p[0].insert(0, p[1])
def p_typedef(self, p):
"""typedef : attributes TYPEDEF type IDENTIFIER ';'"""
p[0] = Typedef(
type=p[3],
name=p[4],
attlist=p[1][
"attlist"],
location=self.getLocation(p, 2),
doccomments=getattr(p[1],
"doccomments", []) + p.slice[2].doccomments,
)
def p_native(self, p):
"""native : attributes NATIVE IDENTIFIER afternativeid '(' NATIVEID ')' ';'"""
p[0] = Native(
name=p[3],
nativename=p[6],
attlist=p[1][
"attlist"],
location=self.getLocation(p, 2),
)
def p_afternativeid(self, p):
"""afternativeid :"""
# this is a place marker: we switch the lexer into literal identifier
# mode here, to slurp up everything until the closeparen
self.lexer.begin(
"nativeid")
def p_webidl(self, p):
"""webidl : WEBIDL IDENTIFIER ';'"""
p[0] = WebIDL(name=p[2], location=self.getLocation(p, 2))
def p_anyident(self, p):
"""anyident : IDENTIFIER
| CONST
"""
p[0] = {
"value": p[1],
"location": self.getLocation(p, 1)}
def p_attributes(self, p):
"""attributes : '[' attlist ']'
|
"""
if len(p) == 1:
p[0] = {
"attlist": []}
else:
p[0] = {
"attlist": p[2],
"doccomments": p.slice[1].doccomments}
def p_attlist_start(self, p):
"""attlist : attribute"""
p[0] = [p[1]]
def p_attlist_continue(self, p):
"""attlist : attribute ',' attlist"""
p[0] = list(p[3])
p[0].insert(0, p[1])
def p_attribute(self, p):
"""attribute : anyident attributeval"""
p[0] = (p[1][
"value"], p[2], p[1][
"location"])
def p_attributeval(self, p):
"""attributeval : '(' IDENTIFIER ')'
--> --------------------
--> maximum size reached
--> --------------------