blob: a161fbfe51d5300b4c1544c3656ed3a79ef34ad4 [file] [log] [blame]
Darin Petkov63138a92012-02-06 14:09:15 +01001// Copyright (c) 2012 The Chromium OS Authors. All rights reserved.
Chris Masoned0ceb8c2011-06-02 10:05:39 -07002// Use of this source code is governed by a BSD-style license that can be
3// found in the LICENSE file.
4
Chris Masone8fe2c7e2011-06-09 15:51:19 -07005#include <map>
Chris Masoned0ceb8c2011-06-02 10:05:39 -07006#include <string>
Chris Masone8fe2c7e2011-06-09 15:51:19 -07007#include <vector>
Chris Masoned0ceb8c2011-06-02 10:05:39 -07008
Chris Masone8fe2c7e2011-06-09 15:51:19 -07009#include <base/logging.h>
Chris Masoned0ceb8c2011-06-02 10:05:39 -070010#include <dbus-c++/dbus.h>
11
Chris Masone889666b2011-07-03 12:58:50 -070012#include "shill/accessor_interface.h"
Chris Masoned0ceb8c2011-06-02 10:05:39 -070013#include "shill/dbus_adaptor.h"
Chris Masone8fe2c7e2011-06-09 15:51:19 -070014#include "shill/error.h"
mukesh agrawal7a4e4002011-09-06 11:26:05 -070015#include "shill/key_value_store.h"
Chris Masoneb925cc82011-06-22 15:39:57 -070016#include "shill/property_store.h"
Chris Masone8fe2c7e2011-06-09 15:51:19 -070017
18using std::map;
19using std::string;
20using std::vector;
Chris Masoned0ceb8c2011-06-02 10:05:39 -070021
22namespace shill {
23
Chris Masone8fe2c7e2011-06-09 15:51:19 -070024// static
Paul Stewartced6a0b2011-11-08 15:32:04 -080025const char DBusAdaptor::kByteArraysSig[] = "aay";
26// static
mukesh agrawal32399322011-09-01 10:53:43 -070027const char DBusAdaptor::kPathArraySig[] = "ao";
28// static
Chris Masone8fe2c7e2011-06-09 15:51:19 -070029const char DBusAdaptor::kStringmapSig[] = "a{ss}";
30// static
Chris Masone27c4aa52011-07-02 13:10:14 -070031const char DBusAdaptor::kStringmapsSig[] = "aa{ss}";
32// static
Chris Masone8fe2c7e2011-06-09 15:51:19 -070033const char DBusAdaptor::kStringsSig[] = "as";
34
35DBusAdaptor::DBusAdaptor(DBus::Connection* conn, const string &object_path)
Chris Masoned0ceb8c2011-06-02 10:05:39 -070036 : DBus::ObjectAdaptor(*conn, object_path) {
Darin Petkov67d8ecf2011-07-26 16:03:30 -070037 VLOG(2) << "DBusAdaptor: " << object_path;
Chris Masoned0ceb8c2011-06-02 10:05:39 -070038}
39
40DBusAdaptor::~DBusAdaptor() {}
41
42// static
mukesh agrawal6bb9e7c2012-01-30 14:57:54 -080043bool DBusAdaptor::SetProperty(PropertyStore *store,
44 const string &name,
45 const ::DBus::Variant &value,
46 ::DBus::Error *error) {
mukesh agrawalffa3d042011-10-06 15:26:10 -070047 Error e;
Chris Masone8fe2c7e2011-06-09 15:51:19 -070048
49 if (DBusAdaptor::IsBool(value.signature()))
mukesh agrawalffa3d042011-10-06 15:26:10 -070050 store->SetBoolProperty(name, value.reader().get_bool(), &e);
Chris Masone8fe2c7e2011-06-09 15:51:19 -070051 else if (DBusAdaptor::IsByte(value.signature()))
mukesh agrawalffa3d042011-10-06 15:26:10 -070052 store->SetUint8Property(name, value.reader().get_byte(), &e);
Chris Masone8fe2c7e2011-06-09 15:51:19 -070053 else if (DBusAdaptor::IsInt16(value.signature()))
mukesh agrawalffa3d042011-10-06 15:26:10 -070054 store->SetInt16Property(name, value.reader().get_int16(), &e);
Chris Masone8fe2c7e2011-06-09 15:51:19 -070055 else if (DBusAdaptor::IsInt32(value.signature()))
mukesh agrawalffa3d042011-10-06 15:26:10 -070056 store->SetInt32Property(name, value.reader().get_int32(), &e);
Chris Masone3bd3c8c2011-06-13 08:20:26 -070057 else if (DBusAdaptor::IsPath(value.signature()))
mukesh agrawalffa3d042011-10-06 15:26:10 -070058 store->SetStringProperty(name, value.reader().get_path(), &e);
Chris Masone8fe2c7e2011-06-09 15:51:19 -070059 else if (DBusAdaptor::IsString(value.signature()))
mukesh agrawalffa3d042011-10-06 15:26:10 -070060 store->SetStringProperty(name, value.reader().get_string(), &e);
Chris Masone8fe2c7e2011-06-09 15:51:19 -070061 else if (DBusAdaptor::IsStringmap(value.signature()))
mukesh agrawalffa3d042011-10-06 15:26:10 -070062 store->SetStringmapProperty(name,
63 value.operator map<string, string>(),
64 &e);
65 else if (DBusAdaptor::IsStringmaps(value.signature())) {
Chris Masone27c4aa52011-07-02 13:10:14 -070066 VLOG(1) << " can't yet handle setting type " << value.signature();
mukesh agrawalffa3d042011-10-06 15:26:10 -070067 e.Populate(Error::kInternalError);
68 } else if (DBusAdaptor::IsStrings(value.signature()))
69 store->SetStringsProperty(name, value.operator vector<string>(), &e);
Chris Masone8fe2c7e2011-06-09 15:51:19 -070070 else if (DBusAdaptor::IsUint16(value.signature()))
mukesh agrawalffa3d042011-10-06 15:26:10 -070071 store->SetUint16Property(name, value.reader().get_uint16(), &e);
Chris Masone8fe2c7e2011-06-09 15:51:19 -070072 else if (DBusAdaptor::IsUint32(value.signature()))
mukesh agrawalffa3d042011-10-06 15:26:10 -070073 store->SetUint32Property(name, value.reader().get_uint32(), &e);
Eric Shienbroodb23d4b92012-02-16 12:32:42 -050074 else if (DBusAdaptor::IsKeyValueStore(value.signature())) {
75 VLOG(1) << " can't yet handle setting type " << value.signature();
76 e.Populate(Error::kInternalError);
77 } else {
Chris Masone27c4aa52011-07-02 13:10:14 -070078 NOTREACHED() << " unknown type: " << value.signature();
mukesh agrawalffa3d042011-10-06 15:26:10 -070079 e.Populate(Error::kInternalError);
Chris Masone27c4aa52011-07-02 13:10:14 -070080 }
mukesh agrawalffa3d042011-10-06 15:26:10 -070081
82 if (error != NULL) {
83 e.ToDBusError(error);
84 }
85
86 return e.IsSuccess();
Chris Masone8fe2c7e2011-06-09 15:51:19 -070087}
88
89// static
mukesh agrawalde29fa82011-09-16 16:16:36 -070090bool DBusAdaptor::GetProperties(const PropertyStore &store,
Chris Masonea8a2c252011-06-27 22:16:30 -070091 map<string, ::DBus::Variant> *out,
mukesh agrawal1830fa12011-09-26 14:31:40 -070092 ::DBus::Error */*error*/) {
Gaurav Shah1b7a6162011-11-09 11:41:01 -080093 Error e;
Chris Masonea8a2c252011-06-27 22:16:30 -070094 {
Gaurav Shah1b7a6162011-11-09 11:41:01 -080095 ReadablePropertyConstIterator<bool> it = store.GetBoolPropertiesIter();
Chris Masonea8a2c252011-06-27 22:16:30 -070096 for ( ; !it.AtEnd(); it.Advance())
Gaurav Shah1b7a6162011-11-09 11:41:01 -080097 (*out)[it.Key()] = BoolToVariant(it.Value(&e));
Chris Masonea8a2c252011-06-27 22:16:30 -070098 }
99 {
Gaurav Shah1b7a6162011-11-09 11:41:01 -0800100 ReadablePropertyConstIterator<int16> it = store.GetInt16PropertiesIter();
Chris Masonea8a2c252011-06-27 22:16:30 -0700101 for ( ; !it.AtEnd(); it.Advance())
Gaurav Shah1b7a6162011-11-09 11:41:01 -0800102 (*out)[it.Key()] = Int16ToVariant(it.Value(&e));
Chris Masonea8a2c252011-06-27 22:16:30 -0700103 }
104 {
Gaurav Shah1b7a6162011-11-09 11:41:01 -0800105 ReadablePropertyConstIterator<int32> it = store.GetInt32PropertiesIter();
Chris Masonea8a2c252011-06-27 22:16:30 -0700106 for ( ; !it.AtEnd(); it.Advance())
Gaurav Shah1b7a6162011-11-09 11:41:01 -0800107 (*out)[it.Key()] = Int32ToVariant(it.Value(&e));
Chris Masonea8a2c252011-06-27 22:16:30 -0700108 }
109 {
Darin Petkov63138a92012-02-06 14:09:15 +0100110 ReadablePropertyConstIterator<KeyValueStore> it =
111 store.GetKeyValueStorePropertiesIter();
112 for ( ; !it.AtEnd(); it.Advance())
113 (*out)[it.Key()] = KeyValueStoreToVariant(it.Value(&e));
114 }
115 {
Gaurav Shah1b7a6162011-11-09 11:41:01 -0800116 ReadablePropertyConstIterator<string> it = store.GetStringPropertiesIter();
Chris Masonea8a2c252011-06-27 22:16:30 -0700117 for ( ; !it.AtEnd(); it.Advance())
Gaurav Shah1b7a6162011-11-09 11:41:01 -0800118 (*out)[it.Key()] = StringToVariant(it.Value(&e));
Chris Masonea8a2c252011-06-27 22:16:30 -0700119 }
120 {
Gaurav Shah1b7a6162011-11-09 11:41:01 -0800121 ReadablePropertyConstIterator<Stringmap> it =
122 store.GetStringmapPropertiesIter();
Chris Masonea8a2c252011-06-27 22:16:30 -0700123 for ( ; !it.AtEnd(); it.Advance())
Gaurav Shah1b7a6162011-11-09 11:41:01 -0800124 (*out)[it.Key()]= StringmapToVariant(it.Value(&e));
Chris Masonea8a2c252011-06-27 22:16:30 -0700125 }
126 {
Gaurav Shah1b7a6162011-11-09 11:41:01 -0800127 ReadablePropertyConstIterator<Strings> it =
128 store.GetStringsPropertiesIter();
Chris Masonea8a2c252011-06-27 22:16:30 -0700129 for ( ; !it.AtEnd(); it.Advance())
Gaurav Shah1b7a6162011-11-09 11:41:01 -0800130 (*out)[it.Key()] = StringsToVariant(it.Value(&e));
Chris Masonea8a2c252011-06-27 22:16:30 -0700131 }
132 {
Gaurav Shah1b7a6162011-11-09 11:41:01 -0800133 ReadablePropertyConstIterator<uint8> it = store.GetUint8PropertiesIter();
Chris Masonea8a2c252011-06-27 22:16:30 -0700134 for ( ; !it.AtEnd(); it.Advance())
Gaurav Shah1b7a6162011-11-09 11:41:01 -0800135 (*out)[it.Key()] = ByteToVariant(it.Value(&e));
Chris Masonea8a2c252011-06-27 22:16:30 -0700136 }
137 {
Gaurav Shah1b7a6162011-11-09 11:41:01 -0800138 ReadablePropertyConstIterator<uint16> it = store.GetUint16PropertiesIter();
Chris Masonea8a2c252011-06-27 22:16:30 -0700139 for ( ; !it.AtEnd(); it.Advance())
Gaurav Shah1b7a6162011-11-09 11:41:01 -0800140 (*out)[it.Key()] = Uint16ToVariant(it.Value(&e));
Chris Masonea8a2c252011-06-27 22:16:30 -0700141 }
142 {
Gaurav Shah1b7a6162011-11-09 11:41:01 -0800143 ReadablePropertyConstIterator<uint32> it = store.GetUint32PropertiesIter();
Chris Masonea8a2c252011-06-27 22:16:30 -0700144 for ( ; !it.AtEnd(); it.Advance())
Gaurav Shah1b7a6162011-11-09 11:41:01 -0800145 (*out)[it.Key()] = Uint32ToVariant(it.Value(&e));
Chris Masonea8a2c252011-06-27 22:16:30 -0700146 }
147 return true;
148}
149
150// static
mukesh agrawal8abd2f62012-01-30 14:56:14 -0800151bool DBusAdaptor::ClearProperty(PropertyStore *store,
152 const string &name,
153 ::DBus::Error *error) {
154 Error e;
155 store->ClearProperty(name, &e);
156
157 if (error != NULL) {
158 e.ToDBusError(error);
159 }
160
161 return e.IsSuccess();
162}
163
164// static
mukesh agrawal7a4e4002011-09-06 11:26:05 -0700165void DBusAdaptor::ArgsToKeyValueStore(
166 const map<string, ::DBus::Variant> &args,
167 KeyValueStore *out,
Gaurav Shah1b7a6162011-11-09 11:41:01 -0800168 Error *error) { // TODO(quiche): Should be ::DBus::Error?
mukesh agrawal7a4e4002011-09-06 11:26:05 -0700169 for (map<string, ::DBus::Variant>::const_iterator it = args.begin();
170 it != args.end();
171 ++it) {
172 DBus::type<string> string_type;
173 DBus::type<bool> bool_type;
174
175 if (it->second.signature() == string_type.sig()) {
176 out->SetString(it->first, it->second.reader().get_string());
177 } else if (it->second.signature() == bool_type.sig()) {
178 out->SetBool(it->first, it->second.reader().get_bool());
179 } else {
180 error->Populate(Error::kInternalError);
Gaurav Shah1b7a6162011-11-09 11:41:01 -0800181 return; // Skip remaining args after error.
mukesh agrawal7a4e4002011-09-06 11:26:05 -0700182 }
183 }
184}
185
186// static
Chris Masoned0ceb8c2011-06-02 10:05:39 -0700187::DBus::Variant DBusAdaptor::BoolToVariant(bool value) {
188 ::DBus::Variant v;
189 v.writer().append_bool(value);
190 return v;
191}
192
193// static
Paul Stewartced6a0b2011-11-08 15:32:04 -0800194::DBus::Variant DBusAdaptor::ByteArraysToVariant(const ByteArrays &value) {
195 ::DBus::MessageIter writer;
196 ::DBus::Variant v;
197
Gaurav Shah7ad8e532011-11-11 17:14:49 -0800198
199 // We have to use a local because the operator<< needs a reference
200 // to work on (the lhs) but writer() returns by-value. C++ prohibits
201 // initializing non-const references from a temporary.
202 // So:
203 // v.writer() << value;
204 // would NOT automagically promote the returned value of v.writer() to
205 // a non-const reference (if you think about it, that's almost always not what
206 // you'd want. see: http://gcc.gnu.org/ml/gcc-help/2006-04/msg00075.html).
207 //
208 // One could consider changing writer() to return a reference, but then it
209 // changes writer() semantics as it can not be a const reference. writer()
210 // currently doesn't modify the original object on which it's called.
Paul Stewartced6a0b2011-11-08 15:32:04 -0800211 writer = v.writer();
212 writer << value;
213 return v;
214}
215
216// static
Chris Masone8fe2c7e2011-06-09 15:51:19 -0700217::DBus::Variant DBusAdaptor::ByteToVariant(uint8 value) {
Chris Masoned0ceb8c2011-06-02 10:05:39 -0700218 ::DBus::Variant v;
Chris Masone8fe2c7e2011-06-09 15:51:19 -0700219 v.writer().append_byte(value);
Chris Masoned0ceb8c2011-06-02 10:05:39 -0700220 return v;
221}
222
223// static
Chris Masone8fe2c7e2011-06-09 15:51:19 -0700224::DBus::Variant DBusAdaptor::Int16ToVariant(int16 value) {
225 ::DBus::Variant v;
226 v.writer().append_int16(value);
227 return v;
228}
229
230// static
231::DBus::Variant DBusAdaptor::Int32ToVariant(int32 value) {
Chris Masoned0ceb8c2011-06-02 10:05:39 -0700232 ::DBus::Variant v;
233 v.writer().append_int32(value);
234 return v;
235}
236
237// static
Chris Masone3bd3c8c2011-06-13 08:20:26 -0700238::DBus::Variant DBusAdaptor::PathToVariant(const ::DBus::Path &value) {
239 ::DBus::Variant v;
240 v.writer().append_path(value.c_str());
241 return v;
242}
243
244// static
mukesh agrawal32399322011-09-01 10:53:43 -0700245::DBus::Variant DBusAdaptor::PathArrayToVariant(
246 const vector< ::DBus::Path> &value) {
247 ::DBus::MessageIter writer;
248 ::DBus::Variant v;
249
Gaurav Shah7ad8e532011-11-11 17:14:49 -0800250 // See note above on why we need to use a local.
mukesh agrawal32399322011-09-01 10:53:43 -0700251 writer = v.writer();
252 writer << value;
253 return v;
254}
255
256// static
Chris Masone8fe2c7e2011-06-09 15:51:19 -0700257::DBus::Variant DBusAdaptor::StringToVariant(const string &value) {
Chris Masoned0ceb8c2011-06-02 10:05:39 -0700258 ::DBus::Variant v;
259 v.writer().append_string(value.c_str());
260 return v;
261}
262
Chris Masone8fe2c7e2011-06-09 15:51:19 -0700263// static
Chris Masone889666b2011-07-03 12:58:50 -0700264::DBus::Variant DBusAdaptor::StringmapToVariant(const Stringmap &value) {
Chris Masone8fe2c7e2011-06-09 15:51:19 -0700265 ::DBus::Variant v;
266 ::DBus::MessageIter writer = v.writer();
267 writer << value;
268 return v;
269}
270
271// static
Chris Masone889666b2011-07-03 12:58:50 -0700272::DBus::Variant DBusAdaptor::StringmapsToVariant(const Stringmaps &value) {
Chris Masoneb925cc82011-06-22 15:39:57 -0700273 ::DBus::Variant v;
274 ::DBus::MessageIter writer = v.writer();
275 writer << value;
276 return v;
277}
278
279// static
Chris Masone889666b2011-07-03 12:58:50 -0700280::DBus::Variant DBusAdaptor::StringsToVariant(const Strings &value) {
Chris Masone8fe2c7e2011-06-09 15:51:19 -0700281 ::DBus::Variant v;
282 ::DBus::MessageIter writer = v.writer();
283 writer << value;
284 return v;
285}
286
287// static
Darin Petkov63138a92012-02-06 14:09:15 +0100288::DBus::Variant DBusAdaptor::KeyValueStoreToVariant(
289 const KeyValueStore &value) {
Chris Masone889666b2011-07-03 12:58:50 -0700290 ::DBus::Variant v;
291 ::DBus::MessageIter writer = v.writer();
Eric Shienbroodb23d4b92012-02-16 12:32:42 -0500292 map<string, ::DBus::Variant> props;
293 {
294 map<string, string>::const_iterator it;
295 for (it = value.string_properties().begin();
296 it != value.string_properties().end();
297 ++it) {
298 ::DBus::Variant vv;
299 ::DBus::MessageIter writer = vv.writer();
300 writer.append_string(it->second.c_str());
301 props[it->first] = vv;
302 }
303 }
304 {
305 map<string, bool>::const_iterator it;
306 for (it = value.bool_properties().begin();
307 it != value.bool_properties().end();
308 ++it) {
309 ::DBus::Variant vv;
310 ::DBus::MessageIter writer = vv.writer();
311 writer.append_bool(it->second);
312 props[it->first] = vv;
313 }
314 }
315 {
316 map<string, uint32>::const_iterator it;
317 for (it = value.uint_properties().begin();
318 it != value.uint_properties().end();
319 ++it) {
320 ::DBus::Variant vv;
321 ::DBus::MessageIter writer = vv.writer();
322 writer.append_uint32(it->second);
323 props[it->first] = vv;
324 }
325 }
326 writer << props;
Chris Masone889666b2011-07-03 12:58:50 -0700327 return v;
328}
329
330// static
Chris Masone8fe2c7e2011-06-09 15:51:19 -0700331::DBus::Variant DBusAdaptor::Uint16ToVariant(uint16 value) {
332 ::DBus::Variant v;
333 v.writer().append_uint16(value);
334 return v;
335}
336
337// static
338::DBus::Variant DBusAdaptor::Uint32ToVariant(uint32 value) {
339 ::DBus::Variant v;
340 v.writer().append_uint32(value);
341 return v;
342}
343
344// static
345bool DBusAdaptor::IsBool(::DBus::Signature signature) {
346 return signature == ::DBus::type<bool>::sig();
347}
348
349// static
350bool DBusAdaptor::IsByte(::DBus::Signature signature) {
351 return signature == ::DBus::type<uint8>::sig();
352}
353
354// static
Paul Stewartced6a0b2011-11-08 15:32:04 -0800355bool DBusAdaptor::IsByteArrays(::DBus::Signature signature) {
356 return signature == DBusAdaptor::kByteArraysSig;
357}
358
359// static
Chris Masone8fe2c7e2011-06-09 15:51:19 -0700360bool DBusAdaptor::IsInt16(::DBus::Signature signature) {
361 return signature == ::DBus::type<int16>::sig();
362}
363
364// static
365bool DBusAdaptor::IsInt32(::DBus::Signature signature) {
366 return signature == ::DBus::type<int32>::sig();
367}
368
369// static
Chris Masone3bd3c8c2011-06-13 08:20:26 -0700370bool DBusAdaptor::IsPath(::DBus::Signature signature) {
371 return signature == ::DBus::type< ::DBus::Path >::sig();
372}
373
374// static
mukesh agrawal32399322011-09-01 10:53:43 -0700375bool DBusAdaptor::IsPathArray(::DBus::Signature signature) {
376 return signature == DBusAdaptor::kPathArraySig;
377}
378
379// static
Chris Masone8fe2c7e2011-06-09 15:51:19 -0700380bool DBusAdaptor::IsString(::DBus::Signature signature) {
381 return signature == ::DBus::type<string>::sig();
382}
383
384// static
385bool DBusAdaptor::IsStringmap(::DBus::Signature signature) {
386 return signature == DBusAdaptor::kStringmapSig;
387}
388
389// static
Chris Masone27c4aa52011-07-02 13:10:14 -0700390bool DBusAdaptor::IsStringmaps(::DBus::Signature signature) {
391 return signature == DBusAdaptor::kStringmapsSig;
392}
393
394// static
Chris Masone8fe2c7e2011-06-09 15:51:19 -0700395bool DBusAdaptor::IsStrings(::DBus::Signature signature) {
396 return signature == DBusAdaptor::kStringsSig;
397}
398
399// static
400bool DBusAdaptor::IsUint16(::DBus::Signature signature) {
401 return signature == ::DBus::type<uint16>::sig();
402}
403
404// static
405bool DBusAdaptor::IsUint32(::DBus::Signature signature) {
406 return signature == ::DBus::type<uint32>::sig();
407}
408
Darin Petkove5bc2cb2011-12-07 14:47:32 +0100409// static
Eric Shienbroodb23d4b92012-02-16 12:32:42 -0500410bool DBusAdaptor::IsKeyValueStore(::DBus::Signature signature) {
411 return signature == ::DBus::type<map<string, ::DBus::Variant> >::sig();
412}
413
414// static
Darin Petkove5bc2cb2011-12-07 14:47:32 +0100415DBusAdaptor::Returner *DBusAdaptor::Returner::Create(DBusAdaptor *adaptor) {
416 return new Returner(adaptor);
417}
418
419DBusAdaptor::Returner::Returner(DBusAdaptor *adaptor)
420 : adaptor_(adaptor),
421 state_(kStateInitialized) {
422 VLOG(2) << __func__ << " @ " << this;
423}
424
425DBusAdaptor::Returner::~Returner() {
426 CHECK(state_ != kStateDestroyed);
427 VLOG(2) << "Destroying returner @ " << this << " state: " << state_;
428 adaptor_ = NULL;
429 state_ = kStateDestroyed;
430}
431
432void DBusAdaptor::Returner::Return() {
433 VLOG(2) << __func__ << " @ " << this << " state: " << state_;
434 switch (state_) {
435 case kStateInitialized:
436 // Service method is returning right away, without any continuation.
437 state_ = kStateReturned;
438 return;
439 case kStateDelayed: {
440 // This return happens in the continuation.
441 DBus::ObjectAdaptor::Continuation *cont =
442 adaptor_->find_continuation(this);
443 CHECK(cont);
444 adaptor_->return_now(cont);
445 delete this;
446 return;
447 }
448 default:
449 NOTREACHED() << "Unexpected state: " << state_;
450 break;
451 }
452}
453
454void DBusAdaptor::Returner::ReturnError(const Error &error) {
455 VLOG(2) << __func__ << " @ " << this << " state: " << state_;
456 switch (state_) {
457 case kStateInitialized:
458 // Service method is returning right away, without any continuation.
459 error_.CopyFrom(error);
460 state_ = kStateReturned;
461 return;
462 case kStateDelayed: {
463 // This return happens in the continuation.
464 DBus::Error dbus_error;
465 error.ToDBusError(&dbus_error);
466 DBus::ObjectAdaptor::Continuation *cont =
467 adaptor_->find_continuation(this);
468 CHECK(cont);
469 adaptor_->return_error(cont, dbus_error);
470 delete this;
471 return;
472 }
473 default:
474 NOTREACHED() << "Unexpected state: " << state_;
475 break;
476 }
477}
478
479void DBusAdaptor::Returner::DelayOrReturn(DBus::Error *error) {
480 VLOG(2) << __func__ << " @ " << this << " state: " << state_;
481 switch (state_) {
482 case kStateInitialized:
483 // Service method needs continuation so delay the return.
484 state_ = kStateDelayed;
485
486 // return_later does not return. It unwinds the stack up to the dbus-c++
487 // message handler by throwing an exception.
488 adaptor_->return_later(this);
489 return;
490 case kStateReturned:
491 // Service method has returned right away, without any continuation.
492 error_.ToDBusError(error);
493 delete this;
494 return;
495 default:
496 NOTREACHED() << "Unexpected state: " << state_;
497 break;
498 }
499}
500
Chris Masoned0ceb8c2011-06-02 10:05:39 -0700501} // namespace shill