| OLD | NEW |
| 1 // Copyright 2014 The Crashpad Authors. All rights reserved. | 1 // Copyright 2014 The Crashpad Authors. All rights reserved. |
| 2 // | 2 // |
| 3 // Licensed under the Apache License, Version 2.0 (the "License"); | 3 // Licensed under the Apache License, Version 2.0 (the "License"); |
| 4 // you may not use this file except in compliance with the License. | 4 // you may not use this file except in compliance with the License. |
| 5 // You may obtain a copy of the License at | 5 // You may obtain a copy of the License at |
| 6 // | 6 // |
| 7 // http://www.apache.org/licenses/LICENSE-2.0 | 7 // http://www.apache.org/licenses/LICENSE-2.0 |
| 8 // | 8 // |
| 9 // Unless required by applicable law or agreed to in writing, software | 9 // Unless required by applicable law or agreed to in writing, software |
| 10 // distributed under the License is distributed on an "AS IS" BASIS, | 10 // distributed under the License is distributed on an "AS IS" BASIS, |
| 11 // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. | 11 // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. |
| 12 // See the License for the specific language governing permissions and | 12 // See the License for the specific language governing permissions and |
| 13 // limitations under the License. | 13 // limitations under the License. |
| 14 | 14 |
| 15 #include "handler/mac/exception_handler_server.h" | 15 #include "handler/mac/exception_handler_server.h" |
| 16 | 16 |
| 17 #include "base/logging.h" | 17 #include "base/logging.h" |
| 18 #include "base/mac/mach_logging.h" | 18 #include "base/mac/mach_logging.h" |
| 19 #include "util/mach/composite_mach_message_server.h" | 19 #include "util/mach/composite_mach_message_server.h" |
| 20 #include "util/mach/mach_extensions.h" | 20 #include "util/mach/mach_extensions.h" |
| 21 #include "util/mach/mach_message.h" | 21 #include "util/mach/mach_message.h" |
| 22 #include "util/mach/mach_message_server.h" | 22 #include "util/mach/mach_message_server.h" |
| 23 #include "util/mach/notify_server.h" | 23 #include "util/mach/notify_server.h" |
| 24 | 24 |
| 25 namespace crashpad { | 25 namespace crashpad { |
| 26 | 26 |
| 27 namespace { | 27 namespace { |
| 28 | 28 |
| 29 class ExceptionHandlerServerRun : public UniversalMachExcServer::Interface, | 29 class ExceptionHandlerServerRun : public UniversalMachExcServer::Interface, |
| 30 public NotifyServer::Interface { | 30 public NotifyServer::DefaultInterface { |
| 31 public: | 31 public: |
| 32 ExceptionHandlerServerRun( | 32 ExceptionHandlerServerRun( |
| 33 mach_port_t exception_port, | 33 mach_port_t exception_port, |
| 34 UniversalMachExcServer::Interface* exception_interface) | 34 UniversalMachExcServer::Interface* exception_interface) |
| 35 : UniversalMachExcServer::Interface(), | 35 : UniversalMachExcServer::Interface(), |
| 36 NotifyServer::Interface(), | 36 NotifyServer::DefaultInterface(), |
| 37 mach_exc_server_(this), | 37 mach_exc_server_(this), |
| 38 notify_server_(this), | 38 notify_server_(this), |
| 39 composite_mach_message_server_(), | 39 composite_mach_message_server_(), |
| 40 exception_interface_(exception_interface), | 40 exception_interface_(exception_interface), |
| 41 exception_port_(exception_port), | 41 exception_port_(exception_port), |
| 42 notify_port_(NewMachPort(MACH_PORT_RIGHT_RECEIVE)), | 42 notify_port_(NewMachPort(MACH_PORT_RIGHT_RECEIVE)), |
| 43 running_(true) { | 43 running_(true) { |
| 44 CHECK(notify_port_.is_valid()); | 44 CHECK(notify_port_.is_valid()); |
| 45 | 45 |
| 46 composite_mach_message_server_.AddHandler(&mach_exc_server_); | 46 composite_mach_message_server_.AddHandler(&mach_exc_server_); |
| (...skipping 90 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
| 137 code_count, | 137 code_count, |
| 138 flavor, | 138 flavor, |
| 139 old_state, | 139 old_state, |
| 140 old_state_count, | 140 old_state_count, |
| 141 new_state, | 141 new_state, |
| 142 new_state_count, | 142 new_state_count, |
| 143 trailer, | 143 trailer, |
| 144 destroy_complex_request); | 144 destroy_complex_request); |
| 145 } | 145 } |
| 146 | 146 |
| 147 // NotifyServer::Interface: | 147 // NotifyServer::DefaultInterface: |
| 148 | |
| 149 kern_return_t DoMachNotifyPortDeleted( | |
| 150 notify_port_t notify, | |
| 151 mach_port_name_t name, | |
| 152 const mach_msg_trailer_t* trailer) override { | |
| 153 return UnimplementedNotifyRoutine(notify); | |
| 154 } | |
| 155 | |
| 156 kern_return_t DoMachNotifyPortDestroyed(notify_port_t notify, | |
| 157 mach_port_t rights, | |
| 158 const mach_msg_trailer_t* trailer, | |
| 159 bool* destroy_request) override { | |
| 160 *destroy_request = true; | |
| 161 return UnimplementedNotifyRoutine(notify); | |
| 162 } | |
| 163 | 148 |
| 164 kern_return_t DoMachNotifyNoSenders( | 149 kern_return_t DoMachNotifyNoSenders( |
| 165 notify_port_t notify, | 150 notify_port_t notify, |
| 166 mach_port_mscount_t mscount, | 151 mach_port_mscount_t mscount, |
| 167 const mach_msg_trailer_t* trailer) override { | 152 const mach_msg_trailer_t* trailer) override { |
| 168 if (notify != notify_port_) { | 153 if (notify != notify_port_) { |
| 169 // The message was received as part of a port set. This check ensures that | 154 // The message was received as part of a port set. This check ensures that |
| 170 // only the authorized sender of the no-senders notification is able to | 155 // only the authorized sender of the no-senders notification is able to |
| 171 // stop the exception server. Otherwise, a malicious client would be able | 156 // stop the exception server. Otherwise, a malicious client would be able |
| 172 // to craft and send a no-senders notification via its exception port, and | 157 // to craft and send a no-senders notification via its exception port, and |
| 173 // cause the handler to stop processing exceptions and exit. | 158 // cause the handler to stop processing exceptions and exit. |
| 174 LOG(WARNING) << "notify port mismatch"; | 159 LOG(WARNING) << "notify port mismatch"; |
| 175 return KERN_FAILURE; | 160 return KERN_FAILURE; |
| 176 } | 161 } |
| 177 | 162 |
| 178 running_ = false; | 163 running_ = false; |
| 179 | 164 |
| 180 return KERN_SUCCESS; | 165 return KERN_SUCCESS; |
| 181 } | 166 } |
| 182 | 167 |
| 183 kern_return_t DoMachNotifySendOnce( | |
| 184 notify_port_t notify, | |
| 185 const mach_msg_trailer_t* trailer) override { | |
| 186 return UnimplementedNotifyRoutine(notify); | |
| 187 } | |
| 188 | |
| 189 kern_return_t DoMachNotifyDeadName( | |
| 190 notify_port_t notify, | |
| 191 mach_port_name_t name, | |
| 192 const mach_msg_trailer_t* trailer) override { | |
| 193 return UnimplementedNotifyRoutine(notify); | |
| 194 } | |
| 195 | |
| 196 private: | 168 private: |
| 197 kern_return_t UnimplementedNotifyRoutine(notify_port_t notify) { | |
| 198 // Most of the routines in the notify subsystem are not expected to be | |
| 199 // called. | |
| 200 if (notify != notify_port_) { | |
| 201 LOG(WARNING) << "notify port mismatch"; | |
| 202 return KERN_FAILURE; | |
| 203 } | |
| 204 | |
| 205 NOTREACHED(); | |
| 206 return MIG_BAD_ID; | |
| 207 } | |
| 208 | |
| 209 UniversalMachExcServer mach_exc_server_; | 169 UniversalMachExcServer mach_exc_server_; |
| 210 NotifyServer notify_server_; | 170 NotifyServer notify_server_; |
| 211 CompositeMachMessageServer composite_mach_message_server_; | 171 CompositeMachMessageServer composite_mach_message_server_; |
| 212 UniversalMachExcServer::Interface* exception_interface_; // weak | 172 UniversalMachExcServer::Interface* exception_interface_; // weak |
| 213 mach_port_t exception_port_; // weak | 173 mach_port_t exception_port_; // weak |
| 214 base::mac::ScopedMachReceiveRight notify_port_; | 174 base::mac::ScopedMachReceiveRight notify_port_; |
| 215 bool running_; | 175 bool running_; |
| 216 | 176 |
| 217 DISALLOW_COPY_AND_ASSIGN(ExceptionHandlerServerRun); | 177 DISALLOW_COPY_AND_ASSIGN(ExceptionHandlerServerRun); |
| 218 }; | 178 }; |
| 219 | 179 |
| 220 } // namespace | 180 } // namespace |
| 221 | 181 |
| 222 ExceptionHandlerServer::ExceptionHandlerServer( | 182 ExceptionHandlerServer::ExceptionHandlerServer( |
| 223 base::mac::ScopedMachReceiveRight receive_port) | 183 base::mac::ScopedMachReceiveRight receive_port) |
| 224 : receive_port_(receive_port.Pass()) { | 184 : receive_port_(receive_port.Pass()) { |
| 225 CHECK(receive_port_.is_valid()); | 185 CHECK(receive_port_.is_valid()); |
| 226 } | 186 } |
| 227 | 187 |
| 228 ExceptionHandlerServer::~ExceptionHandlerServer() { | 188 ExceptionHandlerServer::~ExceptionHandlerServer() { |
| 229 } | 189 } |
| 230 | 190 |
| 231 void ExceptionHandlerServer::Run( | 191 void ExceptionHandlerServer::Run( |
| 232 UniversalMachExcServer::Interface* exception_interface) { | 192 UniversalMachExcServer::Interface* exception_interface) { |
| 233 ExceptionHandlerServerRun run(receive_port_.get(), exception_interface); | 193 ExceptionHandlerServerRun run(receive_port_.get(), exception_interface); |
| 234 run.Run(); | 194 run.Run(); |
| 235 } | 195 } |
| 236 | 196 |
| 237 } // namespace crashpad | 197 } // namespace crashpad |
| OLD | NEW |