// Copyright 2014 The Crashpad Authors
//
// 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/composite_mach_message_server.h"
#include <algorithm>
#include <ostream>
#include <utility>
#include "base/check_op.h"
#include "util/mach/mach_message.h"
namespace crashpad {
CompositeMachMessageServer::CompositeMachMessageServer()
: MachMessageServer::Interface(),
handler_map_(),
request_size_(sizeof(mach_msg_header_t)),
reply_size_(sizeof(mig_reply_error_t)) {
}
CompositeMachMessageServer::~CompositeMachMessageServer() {
}
void CompositeMachMessageServer::AddHandler(
MachMessageServer::Interface* handler) {
// Other cycles would be invalid as well, but they aren’t currently checked.
DCHECK_NE(handler, this);
std::set<mach_msg_id_t> request_ids = handler->MachMessageServerRequestIDs();
for (mach_msg_id_t request_id : request_ids) {
std::pair<HandlerMap::const_iterator, bool> result =
handler_map_.insert(std::make_pair(request_id, handler));
CHECK(result.second) << "duplicate request ID " << request_id;
}
request_size_ =
std::max(request_size_, handler->MachMessageServerRequestSize());
reply_size_ = std::max(reply_size_, handler->MachMessageServerReplySize());
}
bool CompositeMachMessageServer::MachMessageServerFunction(
const mach_msg_header_t* in,
mach_msg_header_t* out,
bool* destroy_complex_request) {
HandlerMap::const_iterator iterator = handler_map_.find(in->msgh_id);
if (iterator == handler_map_.end()) {
// Do what MIG-generated server routines do when they can’t dispatch a
// message.
PrepareMIGReplyFromRequest(in, out);
SetMIGReplyError(out, MIG_BAD_ID);
return false;
}
MachMessageServer::Interface* handler = iterator->second;
return handler->MachMessageServerFunction(in, out, destroy_complex_request);
}
std::set<mach_msg_id_t>
CompositeMachMessageServer::MachMessageServerRequestIDs() {
std::set<mach_msg_id_t> request_ids;
for (const auto& entry : handler_map_) {
request_ids.insert(entry.first);
}
return request_ids;
}
mach_msg_size_t CompositeMachMessageServer::MachMessageServerRequestSize() {
return request_size_;
}
mach_msg_size_t CompositeMachMessageServer::MachMessageServerReplySize() {
return reply_size_;
}
} // namespace crashpad