| {%- import "interface_macros.tmpl" as interface_macros %} |
| {%- set class_name = interface.name %} |
| {%- set proxy_name = interface.name ~ "Proxy" %} |
| {%- set namespace_as_string = "%s"|format(namespace|replace(".","::")) %} |
| |
| {%- macro alloc_params(parameters) %} |
| {%- for param in parameters %} |
| {%- if param.kind|is_object_kind %} |
| {{param.kind|cpp_result_type}} p{{loop.index}}; |
| Deserialize_(params->{{param.name}}.ptr, &p{{loop.index}}); |
| {% endif -%} |
| {%- endfor %} |
| {%- endmacro %} |
| |
| {%- macro pass_params(parameters) %} |
| {%- for param in parameters %} |
| {%- if param.kind|is_string_kind -%} |
| p{{loop.index}} |
| {%- elif param.kind|is_object_kind -%} |
| p{{loop.index}}.Pass() |
| {%- elif param.kind|is_interface_kind -%} |
| mojo::MakeProxy<{{param.kind|get_name_for_kind}}>(mojo::MakeScopedHandle(mojo::internal::FetchAndReset(¶ms->{{param.name}}))) |
| {%- elif param.kind|is_interface_request_kind -%} |
| mojo::MakeRequest<{{param.kind.kind|get_name_for_kind}}>(mojo::MakeScopedHandle(mojo::internal::FetchAndReset(¶ms->{{param.name}}))) |
| {%- elif param.kind|is_any_handle_kind -%} |
| mojo::MakeScopedHandle(mojo::internal::FetchAndReset(¶ms->{{param.name}})) |
| {%- elif param.kind|is_enum_kind -%} |
| static_cast<{{param.kind|cpp_wrapper_type}}>(params->{{param.name}}) |
| {%- else -%} |
| params->{{param.name}} |
| {%- endif -%} |
| {%- if not loop.last %}, {% endif %} |
| {%- endfor %} |
| {%- endmacro %} |
| |
| {%- macro compute_payload_size(params_name, parameters) -%} |
| size_t payload_size = |
| mojo::internal::Align(sizeof({{params_name}})); |
| {#--- Computes #} |
| {%- for param in parameters %} |
| {%- if param.kind|is_object_kind %} |
| payload_size += GetSerializedSize_(in_{{param.name}}); |
| {%- endif %} |
| {%- endfor %} |
| {%- endmacro %} |
| |
| {%- macro build_message(params_name, parameters, params_description) -%} |
| {# TODO(yzshen): Consider refactoring to share code with |
| struct_serialization_definition.tmpl #} |
| {{params_name}}* params = |
| {{params_name}}::New(builder.buffer()); |
| {#--- Sets #} |
| {% for param in parameters %} |
| {%- if param.kind|is_object_kind %} |
| {%- if param.kind|is_array_kind %} |
| mojo::SerializeArray_<{{param.kind|get_array_validate_params|indent(24)}}>( |
| mojo::internal::Forward(in_{{param.name}}), builder.buffer(), ¶ms->{{param.name}}.ptr); |
| {%- elif param.kind|is_map_kind %} |
| mojo::SerializeMap_<{{param.kind.value_kind|get_map_validate_params|indent(24)}}>( |
| mojo::internal::Forward(in_{{param.name}}), builder.buffer(), ¶ms->{{param.name}}.ptr); |
| {%- else %} |
| Serialize_(mojo::internal::Forward(in_{{param.name}}), builder.buffer(), ¶ms->{{param.name}}.ptr); |
| {%- endif %} |
| {%- if not param.kind|is_nullable_kind %} |
| MOJO_INTERNAL_DLOG_SERIALIZATION_WARNING( |
| !params->{{param.name}}.ptr, |
| mojo::internal::VALIDATION_ERROR_UNEXPECTED_NULL_POINTER, |
| "null {{param.name}} argument in {{params_description}}"); |
| {%- endif %} |
| {%- elif param.kind|is_any_handle_kind %} |
| {%- if param.kind|is_interface_kind or |
| param.kind|is_interface_request_kind %} |
| // Delegate handle. |
| params->{{param.name}} = in_{{param.name}}.PassMessagePipe().release(); |
| {%- else %} |
| params->{{param.name}} = in_{{param.name}}.release(); |
| {%- endif %} |
| {%- if not param.kind|is_nullable_kind %} |
| MOJO_INTERNAL_DLOG_SERIALIZATION_WARNING( |
| !params->{{param.name}}.is_valid(), |
| mojo::internal::VALIDATION_ERROR_UNEXPECTED_INVALID_HANDLE, |
| "invalid {{param.name}} argument in {{params_description}}"); |
| {%- endif %} |
| {%- else %} |
| params->{{param.name}} = in_{{param.name}}; |
| {%- endif %} |
| {%- endfor %} |
| mojo::Message message; |
| params->EncodePointersAndHandles(message.mutable_handles()); |
| builder.Finish(&message); |
| {%- endmacro %} |
| |
| {#--- Begin #} |
| const char* {{class_name}}::Name_ = "{{namespace_as_string}}::{{class_name}}"; |
| {#--- Constants #} |
| {% for constant in interface.constants %} |
| const {{constant.kind|cpp_pod_type}} {{interface.name}}::{{constant.name}} = {{constant|constant_value}}; |
| {%- endfor %} |
| |
| {#--- ForwardToCallback definition #} |
| {%- for method in interface.methods -%} |
| {%- if method.response_parameters != None %} |
| class {{class_name}}_{{method.name}}_ForwardToCallback |
| : public mojo::MessageReceiver { |
| public: |
| {{class_name}}_{{method.name}}_ForwardToCallback( |
| const {{interface_macros.declare_callback(method)}}& callback) |
| : callback_(callback) { |
| } |
| virtual bool Accept(mojo::Message* message) override; |
| private: |
| {{interface_macros.declare_callback(method)}} callback_; |
| MOJO_DISALLOW_COPY_AND_ASSIGN({{class_name}}_{{method.name}}_ForwardToCallback); |
| }; |
| bool {{class_name}}_{{method.name}}_ForwardToCallback::Accept( |
| mojo::Message* message) { |
| internal::{{class_name}}_{{method.name}}_ResponseParams_Data* params = |
| reinterpret_cast<internal::{{class_name}}_{{method.name}}_ResponseParams_Data*>( |
| message->mutable_payload()); |
| |
| params->DecodePointersAndHandles(message->mutable_handles()); |
| {{alloc_params(method.response_parameters)|indent(2)}} |
| callback_.Run({{pass_params(method.response_parameters)}}); |
| return true; |
| } |
| {%- endif %} |
| {%- endfor %} |
| |
| {{proxy_name}}::{{proxy_name}}(mojo::MessageReceiverWithResponder* receiver) |
| : receiver_(receiver) { |
| } |
| |
| {#--- Proxy definitions #} |
| |
| {%- for method in interface.methods %} |
| {%- set message_name = |
| "internal::k%s_%s_Name"|format(interface.name, method.name) %} |
| {%- set params_name = |
| "internal::%s_%s_Params_Data"|format(interface.name, method.name) %} |
| {%- set params_description = |
| "%s.%s request"|format(interface.name, method.name) %} |
| void {{proxy_name}}::{{method.name}}( |
| {{interface_macros.declare_request_params("in_", method)}}) { |
| {{compute_payload_size(params_name, method.parameters)}} |
| |
| {%- if method.response_parameters != None %} |
| mojo::internal::RequestMessageBuilder builder({{message_name}}, payload_size); |
| {%- else %} |
| mojo::internal::MessageBuilder builder({{message_name}}, payload_size); |
| {%- endif %} |
| |
| {{build_message(params_name, method.parameters, params_description)}} |
| |
| {%- if method.response_parameters != None %} |
| mojo::MessageReceiver* responder = |
| new {{class_name}}_{{method.name}}_ForwardToCallback(callback); |
| if (!receiver_->AcceptWithResponder(&message, responder)) |
| delete responder; |
| {%- else %} |
| bool ok = receiver_->Accept(&message); |
| // This return value may be ignored as !ok implies the Connector has |
| // encountered an error, which will be visible through other means. |
| MOJO_ALLOW_UNUSED_LOCAL(ok); |
| {%- endif %} |
| } |
| {%- endfor %} |
| |
| {#--- ProxyToResponder definition #} |
| {%- for method in interface.methods -%} |
| {%- if method.response_parameters != None %} |
| {%- set message_name = |
| "internal::k%s_%s_Name"|format(interface.name, method.name) %} |
| {%- set params_name = |
| "internal::%s_%s_ResponseParams_Data"|format(interface.name, method.name) %} |
| {%- set params_description = |
| "%s.%s response"|format(interface.name, method.name) %} |
| class {{class_name}}_{{method.name}}_ProxyToResponder |
| : public {{interface_macros.declare_callback(method)}}::Runnable { |
| public: |
| virtual ~{{class_name}}_{{method.name}}_ProxyToResponder() { |
| delete responder_; |
| } |
| |
| {{class_name}}_{{method.name}}_ProxyToResponder( |
| uint64_t request_id, |
| mojo::MessageReceiver* responder) |
| : request_id_(request_id), |
| responder_(responder) { |
| } |
| |
| virtual void Run({{interface_macros.declare_params("in_", method.response_parameters)}}) const override; |
| |
| private: |
| uint64_t request_id_; |
| mutable mojo::MessageReceiver* responder_; |
| MOJO_DISALLOW_COPY_AND_ASSIGN({{class_name}}_{{method.name}}_ProxyToResponder); |
| }; |
| void {{class_name}}_{{method.name}}_ProxyToResponder::Run( |
| {{interface_macros.declare_params("in_", method.response_parameters)}}) const { |
| {{compute_payload_size(params_name, method.response_parameters)}} |
| mojo::internal::ResponseMessageBuilder builder( |
| {{message_name}}, payload_size, request_id_); |
| {{build_message(params_name, method.response_parameters, params_description)}} |
| bool ok = responder_->Accept(&message); |
| MOJO_ALLOW_UNUSED_LOCAL(ok); |
| // TODO(darin): !ok returned here indicates a malformed message, and that may |
| // be good reason to close the connection. However, we don't have a way to do |
| // that from here. We should add a way. |
| delete responder_; |
| responder_ = nullptr; |
| } |
| {%- endif -%} |
| {%- endfor %} |
| |
| {{class_name}}Stub::{{class_name}}Stub() |
| : sink_(nullptr) { |
| } |
| |
| {#--- Stub definition #} |
| |
| bool {{class_name}}Stub::Accept(mojo::Message* message) { |
| {%- if interface.methods %} |
| switch (message->header()->name) { |
| {%- for method in interface.methods %} |
| case internal::k{{class_name}}_{{method.name}}_Name: { |
| {%- if method.response_parameters == None %} |
| internal::{{class_name}}_{{method.name}}_Params_Data* params = |
| reinterpret_cast<internal::{{class_name}}_{{method.name}}_Params_Data*>( |
| message->mutable_payload()); |
| |
| params->DecodePointersAndHandles(message->mutable_handles()); |
| {{alloc_params(method.parameters)|indent(6)}} |
| // A null |sink_| typically means there is a missing call to |
| // InterfacePtr::set_client(). |
| assert(sink_); |
| sink_->{{method.name}}({{pass_params(method.parameters)}}); |
| return true; |
| {%- else %} |
| break; |
| {%- endif %} |
| } |
| {%- endfor %} |
| } |
| {%- endif %} |
| return false; |
| } |
| |
| bool {{class_name}}Stub::AcceptWithResponder( |
| mojo::Message* message, mojo::MessageReceiver* responder) { |
| {%- if interface.methods %} |
| switch (message->header()->name) { |
| {%- for method in interface.methods %} |
| case internal::k{{class_name}}_{{method.name}}_Name: { |
| {%- if method.response_parameters != None %} |
| internal::{{class_name}}_{{method.name}}_Params_Data* params = |
| reinterpret_cast<internal::{{class_name}}_{{method.name}}_Params_Data*>( |
| message->mutable_payload()); |
| |
| params->DecodePointersAndHandles(message->mutable_handles()); |
| {{interface_macros.declare_callback(method)}}::Runnable* runnable = |
| new {{class_name}}_{{method.name}}_ProxyToResponder( |
| message->request_id(), responder); |
| {{interface_macros.declare_callback(method)}} callback(runnable); |
| {{alloc_params(method.parameters)|indent(6)}} |
| // A null |sink_| typically means there is a missing call to |
| // InterfacePtr::set_client(). |
| assert(sink_); |
| sink_->{{method.name}}( |
| {%- if method.parameters -%}{{pass_params(method.parameters)}}, {% endif -%}callback); |
| return true; |
| {%- else %} |
| break; |
| {%- endif %} |
| } |
| {%- endfor %} |
| } |
| {%- endif %} |
| return false; |
| } |
| |
| {#--- Request validator definitions #} |
| |
| {{class_name}}RequestValidator::{{class_name}}RequestValidator( |
| mojo::MessageReceiver* sink) : MessageFilter(sink) { |
| } |
| |
| bool {{class_name}}RequestValidator::Accept(mojo::Message* message) { |
| {%- if interface.methods %} |
| switch (message->header()->name) { |
| {%- for method in interface.methods %} |
| case internal::k{{class_name}}_{{method.name}}_Name: { |
| {%- if method.response_parameters != None %} |
| if (!message->has_flag(mojo::internal::kMessageExpectsResponse)) |
| break; |
| {%- else %} |
| if (message->has_flag(mojo::internal::kMessageExpectsResponse) || |
| message->has_flag(mojo::internal::kMessageIsResponse)) { |
| break; |
| } |
| {%- endif %} |
| mojo::internal::BoundsChecker bounds_checker( |
| message->payload(), message->payload_num_bytes(), |
| message->handles()->size()); |
| if (!internal::{{class_name}}_{{method.name}}_Params_Data::Validate( |
| message->payload(), &bounds_checker)) { |
| return false; |
| } |
| break; |
| } |
| {%- endfor %} |
| } |
| {%- endif %} |
| |
| // A null |sink_| typically means there is a missing call to |
| // InterfacePtr::set_client(). |
| assert(sink_); |
| return sink_->Accept(message); |
| } |
| |
| {#--- Response validator definitions #} |
| {% if interface|has_callbacks %} |
| {{class_name}}ResponseValidator::{{class_name}}ResponseValidator( |
| mojo::MessageReceiver* sink) : MessageFilter(sink) { |
| } |
| |
| bool {{class_name}}ResponseValidator::Accept(mojo::Message* message) { |
| {%- if interface.methods %} |
| switch (message->header()->name) { |
| {%- for method in interface.methods if method.response_parameters != None %} |
| case internal::k{{class_name}}_{{method.name}}_Name: { |
| if (!message->has_flag(mojo::internal::kMessageIsResponse)) |
| break; |
| mojo::internal::BoundsChecker bounds_checker( |
| message->payload(), message->payload_num_bytes(), |
| message->handles()->size()); |
| if (!internal::{{class_name}}_{{method.name}}_ResponseParams_Data::Validate( |
| message->payload(), &bounds_checker)) { |
| return false; |
| } |
| break; |
| } |
| {%- endfor %} |
| } |
| {%- endif %} |
| |
| // A null |sink_| typically means there is a missing call to |
| // InterfacePtr::set_client(). |
| assert(sink_); |
| return sink_->Accept(message); |
| } |
| {%- endif -%} |