1 // Copyright (c) 2012 The Chromium Authors. All rights reserved.
2 // Use of this source code is governed by a BSD-style license that can be
3 // found in the LICENSE file.
4
5 #include "dbus/values_util.h"
6
7 #include "base/json/json_writer.h"
8 #include "base/logging.h"
9 #include "base/memory/scoped_ptr.h"
10 #include "base/values.h"
11 #include "dbus/message.h"
12
13 namespace dbus {
14
15 namespace {
16
17 // Returns whether |value| is exactly representable by double or not.
18 template<typename T>
IsExactlyRepresentableByDouble(T value)19 bool IsExactlyRepresentableByDouble(T value) {
20 return value == static_cast<T>(static_cast<double>(value));
21 }
22
23 // Pops values from |reader| and appends them to |list_value|.
PopListElements(MessageReader * reader,base::ListValue * list_value)24 bool PopListElements(MessageReader* reader, base::ListValue* list_value) {
25 while (reader->HasMoreData()) {
26 base::Value* element_value = PopDataAsValue(reader);
27 if (!element_value)
28 return false;
29 list_value->Append(element_value);
30 }
31 return true;
32 }
33
34 // Pops dict-entries from |reader| and sets them to |dictionary_value|
PopDictionaryEntries(MessageReader * reader,base::DictionaryValue * dictionary_value)35 bool PopDictionaryEntries(MessageReader* reader,
36 base::DictionaryValue* dictionary_value) {
37 while (reader->HasMoreData()) {
38 DCHECK_EQ(Message::DICT_ENTRY, reader->GetDataType());
39 MessageReader entry_reader(NULL);
40 if (!reader->PopDictEntry(&entry_reader))
41 return false;
42 // Get key as a string.
43 std::string key_string;
44 if (entry_reader.GetDataType() == Message::STRING) {
45 // If the type of keys is STRING, pop it directly.
46 if (!entry_reader.PopString(&key_string))
47 return false;
48 } else {
49 // If the type of keys is not STRING, convert it to string.
50 scoped_ptr<base::Value> key(PopDataAsValue(&entry_reader));
51 if (!key)
52 return false;
53 // Use JSONWriter to convert an arbitrary value to a string.
54 base::JSONWriter::Write(*key, &key_string);
55 }
56 // Get the value and set the key-value pair.
57 base::Value* value = PopDataAsValue(&entry_reader);
58 if (!value)
59 return false;
60 dictionary_value->SetWithoutPathExpansion(key_string, value);
61 }
62 return true;
63 }
64
65 // Gets the D-Bus type signature for the value.
GetTypeSignature(const base::Value & value)66 std::string GetTypeSignature(const base::Value& value) {
67 switch (value.GetType()) {
68 case base::Value::TYPE_BOOLEAN:
69 return "b";
70 case base::Value::TYPE_INTEGER:
71 return "i";
72 case base::Value::TYPE_DOUBLE:
73 return "d";
74 case base::Value::TYPE_STRING:
75 return "s";
76 case base::Value::TYPE_BINARY:
77 return "ay";
78 case base::Value::TYPE_DICTIONARY:
79 return "a{sv}";
80 case base::Value::TYPE_LIST:
81 return "av";
82 default:
83 DLOG(ERROR) << "Unexpected type " << value.GetType();
84 return std::string();
85 }
86 }
87
88 } // namespace
89
PopDataAsValue(MessageReader * reader)90 base::Value* PopDataAsValue(MessageReader* reader) {
91 base::Value* result = NULL;
92 switch (reader->GetDataType()) {
93 case Message::INVALID_DATA:
94 // Do nothing.
95 break;
96 case Message::BYTE: {
97 uint8_t value = 0;
98 if (reader->PopByte(&value))
99 result = new base::FundamentalValue(value);
100 break;
101 }
102 case Message::BOOL: {
103 bool value = false;
104 if (reader->PopBool(&value))
105 result = new base::FundamentalValue(value);
106 break;
107 }
108 case Message::INT16: {
109 int16_t value = 0;
110 if (reader->PopInt16(&value))
111 result = new base::FundamentalValue(value);
112 break;
113 }
114 case Message::UINT16: {
115 uint16_t value = 0;
116 if (reader->PopUint16(&value))
117 result = new base::FundamentalValue(value);
118 break;
119 }
120 case Message::INT32: {
121 int32_t value = 0;
122 if (reader->PopInt32(&value))
123 result = new base::FundamentalValue(value);
124 break;
125 }
126 case Message::UINT32: {
127 uint32_t value = 0;
128 if (reader->PopUint32(&value))
129 result = new base::FundamentalValue(static_cast<double>(value));
130 break;
131 }
132 case Message::INT64: {
133 int64_t value = 0;
134 if (reader->PopInt64(&value)) {
135 DLOG_IF(WARNING, !IsExactlyRepresentableByDouble(value)) <<
136 value << " is not exactly representable by double";
137 result = new base::FundamentalValue(static_cast<double>(value));
138 }
139 break;
140 }
141 case Message::UINT64: {
142 uint64_t value = 0;
143 if (reader->PopUint64(&value)) {
144 DLOG_IF(WARNING, !IsExactlyRepresentableByDouble(value)) <<
145 value << " is not exactly representable by double";
146 result = new base::FundamentalValue(static_cast<double>(value));
147 }
148 break;
149 }
150 case Message::DOUBLE: {
151 double value = 0;
152 if (reader->PopDouble(&value))
153 result = new base::FundamentalValue(value);
154 break;
155 }
156 case Message::STRING: {
157 std::string value;
158 if (reader->PopString(&value))
159 result = new base::StringValue(value);
160 break;
161 }
162 case Message::OBJECT_PATH: {
163 ObjectPath value;
164 if (reader->PopObjectPath(&value))
165 result = new base::StringValue(value.value());
166 break;
167 }
168 case Message::UNIX_FD: {
169 // Cannot distinguish a file descriptor from an int
170 NOTREACHED();
171 break;
172 }
173 case Message::ARRAY: {
174 MessageReader sub_reader(NULL);
175 if (reader->PopArray(&sub_reader)) {
176 // If the type of the array's element is DICT_ENTRY, create a
177 // DictionaryValue, otherwise create a ListValue.
178 if (sub_reader.GetDataType() == Message::DICT_ENTRY) {
179 scoped_ptr<base::DictionaryValue> dictionary_value(
180 new base::DictionaryValue);
181 if (PopDictionaryEntries(&sub_reader, dictionary_value.get()))
182 result = dictionary_value.release();
183 } else {
184 scoped_ptr<base::ListValue> list_value(new base::ListValue);
185 if (PopListElements(&sub_reader, list_value.get()))
186 result = list_value.release();
187 }
188 }
189 break;
190 }
191 case Message::STRUCT: {
192 MessageReader sub_reader(NULL);
193 if (reader->PopStruct(&sub_reader)) {
194 scoped_ptr<base::ListValue> list_value(new base::ListValue);
195 if (PopListElements(&sub_reader, list_value.get()))
196 result = list_value.release();
197 }
198 break;
199 }
200 case Message::DICT_ENTRY:
201 // DICT_ENTRY must be popped as an element of an array.
202 NOTREACHED();
203 break;
204 case Message::VARIANT: {
205 MessageReader sub_reader(NULL);
206 if (reader->PopVariant(&sub_reader))
207 result = PopDataAsValue(&sub_reader);
208 break;
209 }
210 }
211 return result;
212 }
213
AppendBasicTypeValueData(MessageWriter * writer,const base::Value & value)214 void AppendBasicTypeValueData(MessageWriter* writer, const base::Value& value) {
215 switch (value.GetType()) {
216 case base::Value::TYPE_BOOLEAN: {
217 bool bool_value = false;
218 bool success = value.GetAsBoolean(&bool_value);
219 DCHECK(success);
220 writer->AppendBool(bool_value);
221 break;
222 }
223 case base::Value::TYPE_INTEGER: {
224 int int_value = 0;
225 bool success = value.GetAsInteger(&int_value);
226 DCHECK(success);
227 writer->AppendInt32(int_value);
228 break;
229 }
230 case base::Value::TYPE_DOUBLE: {
231 double double_value = 0;
232 bool success = value.GetAsDouble(&double_value);
233 DCHECK(success);
234 writer->AppendDouble(double_value);
235 break;
236 }
237 case base::Value::TYPE_STRING: {
238 std::string string_value;
239 bool success = value.GetAsString(&string_value);
240 DCHECK(success);
241 writer->AppendString(string_value);
242 break;
243 }
244 default:
245 DLOG(ERROR) << "Unexpected type " << value.GetType();
246 break;
247 }
248 }
249
AppendBasicTypeValueDataAsVariant(MessageWriter * writer,const base::Value & value)250 void AppendBasicTypeValueDataAsVariant(MessageWriter* writer,
251 const base::Value& value) {
252 MessageWriter sub_writer(NULL);
253 writer->OpenVariant(GetTypeSignature(value), &sub_writer);
254 AppendBasicTypeValueData(&sub_writer, value);
255 writer->CloseContainer(&sub_writer);
256 }
257
AppendValueData(MessageWriter * writer,const base::Value & value)258 void AppendValueData(MessageWriter* writer, const base::Value& value) {
259 switch (value.GetType()) {
260 case base::Value::TYPE_DICTIONARY: {
261 const base::DictionaryValue* dictionary = NULL;
262 value.GetAsDictionary(&dictionary);
263 dbus::MessageWriter array_writer(NULL);
264 writer->OpenArray("{sv}", &array_writer);
265 for (base::DictionaryValue::Iterator iter(*dictionary);
266 !iter.IsAtEnd(); iter.Advance()) {
267 dbus::MessageWriter dict_entry_writer(NULL);
268 array_writer.OpenDictEntry(&dict_entry_writer);
269 dict_entry_writer.AppendString(iter.key());
270 AppendValueDataAsVariant(&dict_entry_writer, iter.value());
271 array_writer.CloseContainer(&dict_entry_writer);
272 }
273 writer->CloseContainer(&array_writer);
274 break;
275 }
276 case base::Value::TYPE_LIST: {
277 const base::ListValue* list = NULL;
278 value.GetAsList(&list);
279 dbus::MessageWriter array_writer(NULL);
280 writer->OpenArray("v", &array_writer);
281 for (base::ListValue::const_iterator iter = list->begin();
282 iter != list->end(); ++iter) {
283 const base::Value* value = *iter;
284 AppendValueDataAsVariant(&array_writer, *value);
285 }
286 writer->CloseContainer(&array_writer);
287 break;
288 }
289 case base::Value::TYPE_BOOLEAN:
290 case base::Value::TYPE_INTEGER:
291 case base::Value::TYPE_DOUBLE:
292 case base::Value::TYPE_STRING:
293 AppendBasicTypeValueData(writer, value);
294 break;
295 default:
296 DLOG(ERROR) << "Unexpected type: " << value.GetType();
297 }
298 }
299
AppendValueDataAsVariant(MessageWriter * writer,const base::Value & value)300 void AppendValueDataAsVariant(MessageWriter* writer, const base::Value& value) {
301 MessageWriter variant_writer(NULL);
302 writer->OpenVariant(GetTypeSignature(value), &variant_writer);
303 AppendValueData(&variant_writer, value);
304 writer->CloseContainer(&variant_writer);
305 }
306
307 } // namespace dbus
308