#include "edbus_private.h" #include "edbus_private_types.h" static Eina_Bool _compatible_type(int dbus_type, const Eina_Value_Type *value_type) { switch (dbus_type) { case 'i': case 'h': return value_type == EINA_VALUE_TYPE_INT; case 's': case 'o': case 'g': return value_type == EINA_VALUE_TYPE_STRING; case 'b': case 'y': return value_type == EINA_VALUE_TYPE_UCHAR; case 'n': return value_type == EINA_VALUE_TYPE_SHORT; case 'q': return value_type == EINA_VALUE_TYPE_USHORT; case 'u': return value_type == EINA_VALUE_TYPE_UINT; case 'x': return value_type == EINA_VALUE_TYPE_INT64; case 't': return value_type == EINA_VALUE_TYPE_UINT64; case 'd': return value_type == EINA_VALUE_TYPE_DOUBLE; case 'a': return value_type == EINA_VALUE_TYPE_ARRAY; case '(': case '{': case 'e': case 'r': return value_type == EINA_VALUE_TYPE_STRUCT; default: ERR("Unknown type %c", dbus_type); return EINA_FALSE; } } static Eina_Bool _array_append(const char *type, const Eina_Value *value_array, EDBus_Message_Iter *iter) { EDBus_Message_Iter *array; EINA_SAFETY_ON_FALSE_RETURN_VAL( edbus_message_iter_arguments_append(iter, type, &array), EINA_FALSE); DBG("array of type %c", type[1]); switch (type[1]) { case '{': case '(': { unsigned i = strlen(type+2);//remove 'a()' of len a(sv) char *entry_sig = malloc(sizeof(char) * i); memcpy(entry_sig, type+2, i); entry_sig[i-1] = 0; for (i = 0; i < eina_value_array_count(value_array); i++) { Eina_Value st; EDBus_Message_Iter *entry; eina_value_array_value_get(value_array, i, &st); edbus_message_iter_arguments_append(array, type+1, &entry); _message_iter_from_eina_value_struct(entry_sig, entry, &st); edbus_message_iter_container_close(array, entry); eina_value_flush(&st); } free(entry_sig); break; } case 'a': { unsigned i; for (i = 0; i < eina_value_array_count(value_array); i++) { Eina_Value inner_array; EDBus_Message_Iter *sub_array; eina_value_array_value_get(value_array, i, &inner_array); edbus_message_iter_arguments_append(array, type+1, &sub_array); _array_append(type+1, &inner_array, sub_array); edbus_message_iter_container_close(array, sub_array); eina_value_flush(&inner_array); } break; } case 'v': { ERR("Variant not supported."); return EINA_FALSE; } case 'i': case 'h'://fd { int32_t z; unsigned i; for (i = 0; i < eina_value_array_count(value_array); i++) { eina_value_array_get(value_array, i, &z); edbus_message_iter_basic_append(array, type[1], z); } break; } case 's': case 'o'://object path case 'g'://signature { const char *txt; unsigned i; for (i = 0; i < eina_value_array_count(value_array); i++) { eina_value_array_get(value_array, i, &txt); edbus_message_iter_basic_append(array, type[1], txt); } break; } case 'b'://boolean case 'y'://byte { unsigned char z; unsigned i; for (i = 0; i < eina_value_array_count(value_array); i++) { eina_value_array_get(value_array, i, &z); edbus_message_iter_basic_append(array, type[1], z); } break; } case 'n'://int16 { int16_t z; unsigned i; for (i = 0; i < eina_value_array_count(value_array); i++) { eina_value_array_get(value_array, i, &z); edbus_message_iter_basic_append(array, type[1], z); } break; } case 'q'://uint16 { uint16_t z; unsigned i; for (i = 0; i < eina_value_array_count(value_array); i++) { eina_value_array_get(value_array, i, &z); edbus_message_iter_basic_append(array, type[1], z); } break; } case 'u'://uint32 { uint32_t z; unsigned i; for (i = 0; i < eina_value_array_count(value_array); i++) { eina_value_array_get(value_array, i, &z); edbus_message_iter_basic_append(array, type[1], z); } break; } case 'x'://int64 { int64_t z; unsigned i; for (i = 0; i < eina_value_array_count(value_array); i++) { eina_value_array_get(value_array, i, &z); edbus_message_iter_basic_append(array, type[1], z); } break; } case 't'://uint64 { uint64_t z; unsigned i; for (i = 0; i < eina_value_array_count(value_array); i++) { eina_value_array_get(value_array, i, &z); edbus_message_iter_basic_append(array, type[1], z); } break; } case 'd'://double { double z; unsigned i; for (i = 0; i < eina_value_array_count(value_array); i++) { eina_value_array_get(value_array, i, &z); edbus_message_iter_basic_append(array, type[1], z); } break; } default: { ERR("Unknown type %c", type[1]); return EINA_FALSE; } } edbus_message_iter_container_close(iter, array); return EINA_TRUE; } static Eina_Bool _basic_append(char type, const Eina_Value *value, const Eina_Value_Struct_Desc *desc, unsigned idx, EDBus_Message_Iter *iter) { EINA_SAFETY_ON_FALSE_RETURN_VAL( _compatible_type(type, desc->members[idx].type), EINA_FALSE); switch (type) { case 'i'://int case 'h'://fd { int32_t i; eina_value_struct_get(value, desc->members[idx].name, &i); edbus_message_iter_basic_append(iter, type, i); break; } case 's': case 'o'://object path case 'g'://signature { const char *txt; eina_value_struct_get(value, desc->members[idx].name, &txt); edbus_message_iter_basic_append(iter, type, txt); break; } case 'b'://boolean case 'y'://byte { unsigned char byte; eina_value_struct_get(value, desc->members[idx].name, &byte); edbus_message_iter_basic_append(iter, type, byte); break; } case 'n'://int16 { int16_t i; eina_value_struct_get(value, desc->members[idx].name, &i); edbus_message_iter_basic_append(iter, type, i); break; } case 'q'://uint16 { uint16_t i; eina_value_struct_get(value, desc->members[idx].name, &i); edbus_message_iter_basic_append(iter, type, i); break; } case 'u'://uint32 { uint32_t i; eina_value_struct_get(value, desc->members[idx].name, &i); edbus_message_iter_basic_append(iter, type, i); break; } case 'x'://int64 { int64_t i; eina_value_struct_get(value, desc->members[idx].name, &i); edbus_message_iter_basic_append(iter, type, i); break; } case 't'://uint64 { uint64_t i; eina_value_struct_get(value, desc->members[idx].name, &i); edbus_message_iter_basic_append(iter, type, i); break; } case 'd'://double { double d; eina_value_struct_get(value, desc->members[idx].name, &d); edbus_message_iter_basic_append(iter, type, d); break; } default: ERR("Unexpected type %c", type); return EINA_FALSE; } return EINA_TRUE; } Eina_Bool _message_iter_from_eina_value_struct(const char *signature, EDBus_Message_Iter *iter, const Eina_Value *value) { unsigned i; DBusSignatureIter signature_iter; Eina_Bool r = EINA_TRUE; char *type; Eina_Value_Struct st; EINA_SAFETY_ON_FALSE_RETURN_VAL( eina_value_type_get(value) == EINA_VALUE_TYPE_STRUCT, EINA_FALSE); eina_value_pget(value, &st); dbus_signature_iter_init(&signature_iter, signature); i = 0; while ((type = dbus_signature_iter_get_signature(&signature_iter))) { DBG("type: %s", type); if (type[0] != 'v' && !type[1]) r = _basic_append(type[0], value, st.desc, i, iter); else if (type[0] == 'a') { Eina_Value value_array; EINA_SAFETY_ON_FALSE_RETURN_VAL( _compatible_type(type[0], st.desc->members[i].type), EINA_FALSE); eina_value_struct_value_get(value, st.desc->members[i].name, &value_array); r = _array_append(type, &value_array, iter); eina_value_flush(&value_array); } else if (type[0] == '(') { Eina_Value inner_st; EDBus_Message_Iter *sub_iter; char *sub_sig; unsigned len = strlen(type+1) -1; sub_sig = malloc(sizeof(char) * len); memcpy(sub_sig, type+1, len); sub_sig[len-1] = 0; EINA_SAFETY_ON_FALSE_RETURN_VAL( _compatible_type(type[0], st.desc->members[i].type), EINA_FALSE); eina_value_struct_value_get(value, st.desc->members[i].name, &inner_st); edbus_message_iter_arguments_append(iter, type, &sub_iter); r = _message_iter_from_eina_value_struct(sub_sig, sub_iter, &inner_st); edbus_message_iter_container_close(iter, sub_iter); free(sub_sig); } else if (type[0] == 'v') { ERR("Variant not supported"); r = EINA_FALSE; } else { ERR("Unknown type %c", type[0]); r = EINA_FALSE; } i++; dbus_free(type); if (!r || !dbus_signature_iter_next(&signature_iter)) break; } return r; } EAPI Eina_Bool edbus_message_from_eina_value(const char *signature, EDBus_Message *msg, const Eina_Value *value) { EDBus_Message_Iter *iter; EINA_SAFETY_ON_NULL_RETURN_VAL(signature, EINA_FALSE); EINA_SAFETY_ON_NULL_RETURN_VAL(msg, EINA_FALSE); EINA_SAFETY_ON_NULL_RETURN_VAL(value, EINA_FALSE); iter = edbus_message_iter_get(msg); EINA_SAFETY_ON_NULL_RETURN_VAL(iter, EINA_FALSE); EINA_SAFETY_ON_FALSE_RETURN_VAL(iter->writable, EINA_FALSE); return _message_iter_from_eina_value_struct(signature, iter, value); }