Skip to content

Commit

Permalink
Bug 1814683 - Part 4: Allow IPDL structs to contain non-default-const…
Browse files Browse the repository at this point in the history
…ructable types, r=ipc-reviewers,jld

Previously, we would always generate a default constructor for IPDL structs
which explicitly initializes every member. This would require members to have
default initializers statically. With the new approach, each member is
individually wrapped with a template parameter which will try to ensure value
initiaization, and the default constructor is implemented with `= default;`.

The default constructor will produce a warning on clang if it is implicitly
deleted, so that warning is also suppressed.

Differential Revision: https://phabricator.services.mozilla.com/D168881
  • Loading branch information
mystor committed Mar 20, 2023
1 parent 9a282e0 commit fc898b8
Show file tree
Hide file tree
Showing 3 changed files with 106 additions and 55 deletions.
37 changes: 37 additions & 0 deletions ipc/glue/IPDLStructMember.h
Original file line number Diff line number Diff line change
@@ -0,0 +1,37 @@
/* -*- Mode: C++; tab-width: 8; indent-tabs-mode: nil; c-basic-offset: 2 -*- */
/* vim: set ts=8 sts=2 et sw=2 tw=80: */
/* 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/. */

#ifndef mozilla_ipc_IPDLStructMember_h
#define mozilla_ipc_IPDLStructMember_h

#include <type_traits>
#include <utility>
#include "mozilla/Attributes.h"

namespace mozilla::ipc {

// For types which are trivially default constructible, like `int`, force the
// value constructor by wrapping the type in this struct. This is an
// implementation detail which will be hidden by the generated IPDL compiler
// code.
template <typename T>
struct IPDLStructMemberWrapper {
template <typename... Args>
MOZ_IMPLICIT IPDLStructMemberWrapper(Args&&... aArgs)
: mVal(std::forward<Args>(aArgs)...) {}
MOZ_IMPLICIT operator T&() { return mVal; }
MOZ_IMPLICIT operator const T&() const { return mVal; }
T mVal{};
};

template <typename T>
using IPDLStructMember =
std::conditional_t<std::is_trivially_default_constructible_v<T>,
IPDLStructMemberWrapper<T>, T>;

} // namespace mozilla::ipc

#endif // mozilla_ipc_IPDLStructMember_h
1 change: 1 addition & 0 deletions ipc/glue/moz.build
Original file line number Diff line number Diff line change
Expand Up @@ -40,6 +40,7 @@ EXPORTS.mozilla.ipc += [
"IPCStreamUtils.h",
"IPCTypes.h",
"IPDLParamTraits.h",
"IPDLStructMember.h",
"LibrarySandboxPreload.h",
"MessageChannel.h",
"MessageLink.h",
Expand Down
123 changes: 68 additions & 55 deletions ipc/ipdl/ipdl/lower.py
Original file line number Diff line number Diff line change
Expand Up @@ -1488,6 +1488,7 @@ def visitTranslationUnit(self, tu):
for su in tu.structsAndUnions:
typedeps = _ComputeTypeDeps(su.decl.type, typesToIncludes)
if isinstance(su, ipdl.ast.StructDecl):
aggregateTypeIncludes.add("mozilla/ipc/IPDLStructMember.h")
for f in su.fields:
f.ipdltype.accept(typedeps)
elif isinstance(su, ipdl.ast.UnionDecl):
Expand Down Expand Up @@ -2529,64 +2530,43 @@ def _generateCxxStruct(sd):

constreftype = Type(sd.name, const=True, ref=True)

# Struct()
# We want the default constructor to be declared if it is available, but
# some of our members may not be default-constructible. Silence the
# warning which clang generates in that case.
#
# Members which need value initialization will be handled by wrapping
# the member in a template type when declaring them.
struct.addcode(
"""
#ifdef __clang__
# pragma clang diagnostic push
# if __has_warning("-Wdefaulted-function-deleted")
# pragma clang diagnostic ignored "-Wdefaulted-function-deleted"
# endif
#endif
${name}() = default;
#ifdef __clang__
# pragma clang diagnostic pop
#endif
""",
name=sd.name,
)

# If this is an empty struct (no fields), then the default ctor
# and "create-with-fields" ctors are equivalent. So don't bother
# with the default ctor.
# and "create-with-fields" ctors are equivalent.
if len(sd.fields):
assert len(sd.fields) == len(sd.packed_field_order)

# Struct()
defctor = ConstructorDefn(ConstructorDecl(sd.name, force_inline=True))

# We want to explicitly default-construct every member of the struct.
# This will initialize all primitives which wouldn't be initialized
# normally to their default values, and will initialize any actor member
# pointers to the correct default value of `nullptr`. Other C++ types
# with custom constructors must also provide a default constructor.
defctor.memberinits = [
ExprMemberInit(f.memberVar()) for f in sd.fields_member_order()
]
struct.addstmts([defctor, Whitespace.NL])

# Struct(const field1& _f1, ...)
valctor = ConstructorDefn(
ConstructorDecl(
sd.name,
params=[
Decl(
f.forceMoveType()
if _cxxTypeNeedsMoveForData(f.ipdltype)
else f.constRefType(),
f.argVar().name,
)
for f in sd.fields_ipdl_order()
],
force_inline=True,
)
)
valctor.memberinits = []
for f in sd.fields_member_order():
arg = f.argVar()
if _cxxTypeNeedsMoveForData(f.ipdltype):
arg = ExprMove(arg)
valctor.memberinits.append(ExprMemberInit(f.memberVar(), args=[arg]))

struct.addstmts([valctor, Whitespace.NL])

# If a constructor which moves each argument would be different from the
# `const T&` version, also generate that constructor.
if not all(
_cxxTypeNeedsMoveForData(f.ipdltype) or not _cxxTypeCanMove(f.ipdltype)
for f in sd.fields_ipdl_order()
):
# Struct(field1&& _f1, ...)
valmovector = ConstructorDefn(
# Struct(const field1& _f1, ...)
valctor = ConstructorDefn(
ConstructorDecl(
sd.name,
params=[
Decl(
f.forceMoveType()
if _cxxTypeCanMove(f.ipdltype)
if _cxxTypeNeedsMoveForData(f.ipdltype)
else f.constRefType(),
f.argVar().name,
)
Expand All @@ -2595,15 +2575,48 @@ def _generateCxxStruct(sd):
force_inline=True,
)
)

valmovector.memberinits = []
valctor.memberinits = []
for f in sd.fields_member_order():
arg = f.argVar()
if _cxxTypeCanMove(f.ipdltype):
if _cxxTypeNeedsMoveForData(f.ipdltype):
arg = ExprMove(arg)
valmovector.memberinits.append(ExprMemberInit(f.memberVar(), args=[arg]))
valctor.memberinits.append(ExprMemberInit(f.memberVar(), args=[arg]))

struct.addstmts([valctor, Whitespace.NL])

# If a constructor which moves each argument would be different from the
# `const T&` version, also generate that constructor.
if not all(
_cxxTypeNeedsMoveForData(f.ipdltype) or not _cxxTypeCanMove(f.ipdltype)
for f in sd.fields_ipdl_order()
):
# Struct(field1&& _f1, ...)
valmovector = ConstructorDefn(
ConstructorDecl(
sd.name,
params=[
Decl(
f.forceMoveType()
if _cxxTypeCanMove(f.ipdltype)
else f.constRefType(),
f.argVar().name,
)
for f in sd.fields_ipdl_order()
],
force_inline=True,
)
)

valmovector.memberinits = []
for f in sd.fields_member_order():
arg = f.argVar()
if _cxxTypeCanMove(f.ipdltype):
arg = ExprMove(arg)
valmovector.memberinits.append(
ExprMemberInit(f.memberVar(), args=[arg])
)

struct.addstmts([valmovector, Whitespace.NL])
struct.addstmts([valmovector, Whitespace.NL])

# The default copy, move, and assignment constructors, and the default
# destructor, will do the right thing.
Expand Down Expand Up @@ -2697,7 +2710,7 @@ def _effectiveMemberType(f):
# class of CopyableTArray<T>.
if effective_type.name == "nsTArray":
effective_type.name = "CopyableTArray"
return effective_type
return Type("::mozilla::ipc::IPDLStructMember", T=[effective_type])


# --------------------------------------------------
Expand Down

0 comments on commit fc898b8

Please sign in to comment.