Fix obexd crash in 3-way calling scenario. Crash happens when there is redialed second incoming call. Cache for the PBAP session is already created at that moment, but PBAP object is destroyed. Crash happens when object is dereferenced in vobject_list_open. Therefore, PBAP object has to be created before any attempt to write cached data to buffer associated to this object. However, cache_ready_notify function, which is invoked in vobject_vcard_open for valid cache case, sends also PBAP object data via callback function to obex.c and written to OBEX stream as GET response in handle_async_io handler function. A new response is sent to OBEX stream after cache_ready_notify exists to vobject_list_open function, which is callback function for obex_mime_type_driver. Such leads to undefined befavior. Therefore, cache_ready_notify is splitted in two cache_ready_notify and generate_response functions. generate_response fills data to buffer and returns error, if any, while cache_ready_notify notifies OBEX core to write this data to stream. In order to avoid writing to stream twice, cache_ready_notify is replaced by generate_response in vobject_list_open. As a result, PBAP buffer data is generated from existing cache and sent to stream upon start of OBEX stream after vobject_list_open exits. --- plugins/pbap.c | 98 +++++++++++++++++++++++++++++++------------------------- 1 files changed, 54 insertions(+), 44 deletions(-) diff --git a/plugins/pbap.c b/plugins/pbap.c index 1a7d001..4086227 100644 --- a/plugins/pbap.c +++ b/plugins/pbap.c @@ -378,7 +378,7 @@ static GSList *sort_entries(GSList *l, uint8_t order, uint8_t search_attrib, return sorted; } -static void cache_ready_notify(void *user_data) +static int generate_response(void *user_data) { struct pbap_session *pbap = user_data; GSList *sorted; @@ -398,7 +398,8 @@ static void cache_ready_notify(void *user_data) memcpy(hdr->val, &size, sizeof(size)); pbap->obj->buffer = g_string_new_len(aparam, sizeof(aparam)); - goto done; + + return 0; } /* @@ -409,11 +410,8 @@ static void cache_ready_notify(void *user_data) pbap->params->searchattrib, (const char *) pbap->params->searchval); - if (sorted == NULL) { - pbap->cache.valid = TRUE; - obex_object_set_io_flags(pbap->obj, G_IO_ERR, -ENOENT); - return; - } + if (sorted == NULL) + return -ENOENT; /* Computing offset considering first entry of the phonebook */ l = g_slist_nth(sorted, pbap->params->liststartoffset); @@ -430,11 +428,25 @@ static void cache_ready_notify(void *user_data) g_slist_free(sorted); -done: - if (!pbap->cache.valid) { - pbap->cache.valid = TRUE; - obex_object_set_io_flags(pbap->obj, G_IO_IN, 0); + return 0; +} + +static void cache_ready_notify(void *user_data) +{ + struct pbap_session *pbap = user_data; + int err; + + DBG(""); + + pbap->cache.valid = TRUE; + + err = generate_response(pbap); + if (err < 0) { + obex_object_set_io_flags(pbap->obj, G_IO_ERR, err); + return; } + + obex_object_set_io_flags(pbap->obj, G_IO_IN, 0); } static void cache_entry_done(void *user_data) @@ -746,10 +758,28 @@ fail: return NULL; } +static int vobject_close(void *object) +{ + struct pbap_object *obj = object; + + DBG(""); + + if (obj->session) + obj->session->obj = NULL; + + if (obj->buffer) + g_string_free(obj->buffer, TRUE); + + g_free(obj); + + return 0; +} + static void *vobject_list_open(const char *name, int oflag, mode_t mode, void *context, size_t *size, int *err) { struct pbap_session *pbap = context; + struct pbap_object *obj = NULL; int ret; DBG("name %s context %p valid %d", name, context, pbap->cache.valid); @@ -766,30 +796,25 @@ static void *vobject_list_open(const char *name, int oflag, mode_t mode, /* PullvCardListing always get the contacts from the cache */ - if (pbap->cache.valid) { - /* - * Valid cache and empty buffer mean that cache was already - * created within a single session, but no data is available. - */ - if (!pbap->obj->buffer) { - ret = -ENOENT; - goto fail; - } - - cache_ready_notify(pbap); - goto done; - } - - ret = phonebook_create_cache(name, - cache_entry_notify, cache_ready_notify, pbap); + obj = vobject_create(pbap); + if (pbap->cache.valid) + ret = generate_response(pbap); + else + ret = phonebook_create_cache(name, cache_entry_notify, + cache_ready_notify, pbap); if (ret < 0) goto fail; -done: - return vobject_create(pbap); + if (err) + *err = 0; + + return obj; fail: + if (obj) + vobject_close(obj); + if (err) *err = ret; @@ -902,21 +927,6 @@ static ssize_t vobject_vcard_read(void *object, void *buf, size_t count, return string_read(obj->buffer, buf, count); } -static int vobject_close(void *object) -{ - struct pbap_object *obj = object; - - if (obj->session) - obj->session->obj = NULL; - - if (obj->buffer) - g_string_free(obj->buffer, TRUE); - - g_free(obj); - - return 0; -} - static struct obex_mime_type_driver mime_pull = { .target = PBAP_TARGET, .target_size = TARGET_SIZE, -- 1.7.0.4 -- To unsubscribe from this list: send the line "unsubscribe linux-bluetooth" in the body of a message to majordomo@xxxxxxxxxxxxxxx More majordomo info at http://vger.kernel.org/majordomo-info.html