Chromium Code Reviews
chromiumcodereview-hr@appspot.gserviceaccount.com (chromiumcodereview-hr) | Please choose your nickname with Settings | Help | Chromium Project | Gerrit Changes | Sign out
(83)

Unified Diff: util/mach/notify_server.cc

Issue 804633002: Add NotifyServer and its test (Closed) Base URL: https://chromium.googlesource.com/crashpad/crashpad@master
Patch Set: Created 6 years ago
Use n/p to move between diff chunks; N/P to move between comments. Draft comments are only viewable by you.
Jump to:
View side-by-side diff with in-line comments
Download patch
« no previous file with comments | « util/mach/notify_server.h ('k') | util/mach/notify_server_test.cc » ('j') | no next file with comments »
Expand Comments ('e') | Collapse Comments ('c') | Show Comments Hide Comments ('s')
Index: util/mach/notify_server.cc
diff --git a/util/mach/notify_server.cc b/util/mach/notify_server.cc
new file mode 100644
index 0000000000000000000000000000000000000000..05495c65dd3ec9ccbce584921c777f257f2481b4
--- /dev/null
+++ b/util/mach/notify_server.cc
@@ -0,0 +1,245 @@
+// Copyright 2014 The Crashpad Authors. All rights reserved.
+//
+// Licensed under the Apache License, Version 2.0 (the "License");
+// you may not use this file except in compliance with the License.
+// You may obtain a copy of the License at
+//
+// http://www.apache.org/licenses/LICENSE-2.0
+//
+// Unless required by applicable law or agreed to in writing, software
+// distributed under the License is distributed on an "AS IS" BASIS,
+// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+// See the License for the specific language governing permissions and
+// limitations under the License.
+
+#include "util/mach/notify_server.h"
+
+#include "base/logging.h"
+#include "util/mach/notifyServer.h"
+#include "util/mach/mach_message.h"
+
+extern "C" {
+
+// These five functions are not used, and are in fact obsoleted by the other
+// functionality implemented in this file. The standard MIG-generated
+// notify_server() (in notifyServer.c) server dispatch routine usable with the
+// standard mach_msg_server() function calls out to this function.
+// notify_server() is unused and is replaced by the more flexible NotifyServer,
+// but the linker still needs to see these five function definitions.
+
+kern_return_t do_mach_notify_port_deleted(notify_port_t notify,
+ mach_port_name_t name) {
+ NOTREACHED();
+ return KERN_FAILURE;
+}
+
+kern_return_t do_mach_notify_port_destroyed(notify_port_t notify,
+ mach_port_t rights) {
+ NOTREACHED();
+ return KERN_FAILURE;
+}
+
+kern_return_t do_mach_notify_no_senders(notify_port_t notify,
+ mach_port_mscount_t mscount) {
+ NOTREACHED();
+ return KERN_FAILURE;
+}
+
+kern_return_t do_mach_notify_send_once(notify_port_t notify) {
+ NOTREACHED();
+ return KERN_FAILURE;
+}
+
+kern_return_t do_mach_notify_dead_name(notify_port_t notify,
+ mach_port_name_t name) {
+ NOTREACHED();
+ return KERN_FAILURE;
+}
+
+} // extern "C"
+
+namespace {
+
+// The MIG-generated __MIG_check__Request__*() functions are not declared as
+// accepting const data, but they could have been because they in fact do not
+// modify the data. These wrapper functions are provided to bridge the const gap
+// between the code in this file, which is const-correct and treats request
+// message data as const, and the generated functions.
+
+kern_return_t MIGCheckRequestMachNotifyPortDeleted(
+ const __Request__mach_notify_port_deleted_t* in_request) {
+ using Request = __Request__mach_notify_port_deleted_t;
+ return __MIG_check__Request__mach_notify_port_deleted_t(
+ const_cast<Request*>(in_request));
+}
+
+kern_return_t MIGCheckRequestMachNotifyPortDestroyed(
+ const __Request__mach_notify_port_destroyed_t* in_request) {
+ using Request = __Request__mach_notify_port_destroyed_t;
+ return __MIG_check__Request__mach_notify_port_destroyed_t(
+ const_cast<Request*>(in_request));
+}
+
+kern_return_t MIGCheckRequestMachNotifyNoSenders(
+ const __Request__mach_notify_no_senders_t* in_request) {
+ using Request = __Request__mach_notify_no_senders_t;
+ return __MIG_check__Request__mach_notify_no_senders_t(
+ const_cast<Request*>(in_request));
+}
+
+kern_return_t MIGCheckRequestMachNotifySendOnce(
+ const __Request__mach_notify_send_once_t* in_request) {
+ using Request = __Request__mach_notify_send_once_t;
+ return __MIG_check__Request__mach_notify_send_once_t(
+ const_cast<Request*>(in_request));
+}
+
+kern_return_t MIGCheckRequestMachNotifyDeadName(
+ const __Request__mach_notify_dead_name_t* in_request) {
+ using Request = __Request__mach_notify_dead_name_t;
+ return __MIG_check__Request__mach_notify_dead_name_t(
+ const_cast<Request*>(in_request));
+}
+
+} // namespace
+
+namespace crashpad {
+
+NotifyServer::NotifyServer(NotifyServer::Interface* interface)
+ : MachMessageServer::Interface(),
+ interface_(interface) {
+}
+
+bool NotifyServer::MachMessageServerFunction(
+ const mach_msg_header_t* in_header,
+ mach_msg_header_t* out_header,
+ bool* destroy_complex_request) {
+ PrepareMIGReplyFromRequest(in_header, out_header);
+
+ const mach_msg_trailer_t* in_trailer =
+ MachMessageTrailerFromHeader(in_header);
+
+ switch (in_header->msgh_id) {
+ case MACH_NOTIFY_PORT_DELETED: {
+ // mach_notify_port_deleted(), do_mach_notify_port_deleted().
+ using Request = __Request__mach_notify_port_deleted_t;
+ const Request* in_request = reinterpret_cast<const Request*>(in_header);
+ kern_return_t kr = MIGCheckRequestMachNotifyPortDeleted(in_request);
+ if (kr != MACH_MSG_SUCCESS) {
+ SetMIGReplyError(out_header, kr);
+ return true;
+ }
+
+ using Reply = __Reply__mach_notify_port_deleted_t;
+ Reply* out_reply = reinterpret_cast<Reply*>(out_header);
+ out_reply->RetCode =
+ interface_->DoMachNotifyPortDeleted(in_header->msgh_local_port,
+ in_request->name,
+ in_trailer);
+ return true;
+ }
+
+ case MACH_NOTIFY_PORT_DESTROYED: {
+ // mach_notify_port_destroyed(), do_mach_notify_port_destroyed().
+ using Request = __Request__mach_notify_port_destroyed_t;
+ const Request* in_request = reinterpret_cast<const Request*>(in_header);
+ kern_return_t kr = MIGCheckRequestMachNotifyPortDestroyed(in_request);
+ if (kr != MACH_MSG_SUCCESS) {
+ SetMIGReplyError(out_header, kr);
+ return true;
+ }
+
+ using Reply = __Reply__mach_notify_port_destroyed_t;
+ Reply* out_reply = reinterpret_cast<Reply*>(out_header);
+ out_reply->RetCode =
+ interface_->DoMachNotifyPortDestroyed(in_header->msgh_local_port,
+ in_request->rights.name,
+ in_trailer,
+ destroy_complex_request);
+ return true;
+ }
+
+ case MACH_NOTIFY_NO_SENDERS: {
+ // mach_notify_no_senders(), do_mach_notify_no_senders().
+ using Request = __Request__mach_notify_no_senders_t;
+ const Request* in_request = reinterpret_cast<const Request*>(in_header);
+ kern_return_t kr = MIGCheckRequestMachNotifyNoSenders(in_request);
+ if (kr != MACH_MSG_SUCCESS) {
+ SetMIGReplyError(out_header, kr);
+ return true;
+ }
+
+ using Reply = __Reply__mach_notify_no_senders_t;
+ Reply* out_reply = reinterpret_cast<Reply*>(out_header);
+ out_reply->RetCode =
+ interface_->DoMachNotifyNoSenders(in_header->msgh_local_port,
+ in_request->mscount,
+ in_trailer);
+ return true;
+ }
+
+ case MACH_NOTIFY_SEND_ONCE: {
+ // mach_notify_send_once(), do_mach_notify_send_once().
+ using Request = __Request__mach_notify_send_once_t;
+ const Request* in_request = reinterpret_cast<const Request*>(in_header);
+ kern_return_t kr = MIGCheckRequestMachNotifySendOnce(in_request);
+ if (kr != MACH_MSG_SUCCESS) {
+ SetMIGReplyError(out_header, kr);
+ return true;
+ }
+
+ using Reply = __Reply__mach_notify_send_once_t;
+ Reply* out_reply = reinterpret_cast<Reply*>(out_header);
+ out_reply->RetCode =
+ interface_->DoMachNotifySendOnce(in_header->msgh_local_port,
+ in_trailer);
+ return true;
+ }
+
+ case MACH_NOTIFY_DEAD_NAME: {
+ // mach_notify_dead_name(), do_mach_notify_dead_name().
+ using Request = __Request__mach_notify_dead_name_t;
+ const Request* in_request = reinterpret_cast<const Request*>(in_header);
+ kern_return_t kr = MIGCheckRequestMachNotifyDeadName(in_request);
+ if (kr != MACH_MSG_SUCCESS) {
+ SetMIGReplyError(out_header, kr);
+ return true;
+ }
+
+ using Reply = __Reply__mach_notify_dead_name_t;
+ Reply* out_reply = reinterpret_cast<Reply*>(out_header);
+ out_reply->RetCode =
+ interface_->DoMachNotifyDeadName(in_header->msgh_local_port,
+ in_request->name,
+ in_trailer);
+ return true;
+ }
+
+ default: {
+ SetMIGReplyError(out_header, MIG_BAD_ID);
+ return false;
+ }
+ }
+}
+
+std::set<mach_msg_id_t> NotifyServer::MachMessageServerRequestIDs() {
+ const mach_msg_id_t request_ids[] = {
+ MACH_NOTIFY_PORT_DELETED,
+ MACH_NOTIFY_PORT_DESTROYED,
+ MACH_NOTIFY_NO_SENDERS,
+ MACH_NOTIFY_SEND_ONCE,
+ MACH_NOTIFY_DEAD_NAME,
+ };
+ return std::set<mach_msg_id_t>(&request_ids[0],
+ &request_ids[arraysize(request_ids)]);
+}
+
+mach_msg_size_t NotifyServer::MachMessageServerRequestSize() {
+ return sizeof(__RequestUnion__do_notify_subsystem);
+}
+
+mach_msg_size_t NotifyServer::MachMessageServerReplySize() {
+ return sizeof(__ReplyUnion__do_notify_subsystem);
+}
+
+} // namespace crashpad
« no previous file with comments | « util/mach/notify_server.h ('k') | util/mach/notify_server_test.cc » ('j') | no next file with comments »

Powered by Google App Engine
This is Rietveld 408576698