Open Source Telephony
 help / color / mirror / Atom feed
From: Guillaume Zajac <guillaume.zajac@linux.intel.com>
To: ofono@ofono.org
Subject: Re: [PATCH_v7] connman: add plugin in oFono to request/release private network
Date: Wed, 08 Jun 2011 10:46:11 +0200	[thread overview]
Message-ID: <4DEF36D3.5090106@linux.intel.com> (raw)
In-Reply-To: <4DEC70E3.7090604@gmail.com>

[-- Attachment #1: Type: text/plain, Size: 11199 bytes --]

Hi Denis,

On 06/06/2011 08:17, Denis Kenzior wrote:
> Hi Guillaume,
>
> On 06/06/2011 04:17 AM, Guillaume Zajac wrote:
>> Hi,
>>
>> Changelog from v6:
>> 	- This patch contains only the ConnMan plugin to request/release
>> 	the private network settings.
>> 	- remove unused ConnMan interfaces.
>> 	- remove error field from pns_req structure.
>> 	- directly call release method from plugin when a request is redundant.
>>
>> ---
>>   Makefile.am       |    3 +
>>   plugins/connman.c |  277 +++++++++++++++++++++++++++++++++++++++++++++++++++++
>>   2 files changed, 280 insertions(+), 0 deletions(-)
>>   create mode 100644 plugins/connman.c
>>
>> diff --git a/Makefile.am b/Makefile.am
>> index 0a7b6d5..ab28e2c 100644
>> --- a/Makefile.am
>> +++ b/Makefile.am
>> @@ -339,6 +339,9 @@ builtin_sources += plugins/hfp_ag.c plugins/bluetooth.h
>>   builtin_modules += dun_gw
>>   builtin_sources += plugins/dun_gw.c plugins/bluetooth.h
>>
>> +builtin_modules += connman
>> +builtin_sources += plugins/connman.c
>> +
>>   builtin_sources += $(btio_sources)
>>   builtin_cflags += @BLUEZ_CFLAGS@
>>   builtin_libadd += @BLUEZ_LIBS@
>> diff --git a/plugins/connman.c b/plugins/connman.c
>> new file mode 100644
>> index 0000000..e6cce3f
>> --- /dev/null
>> +++ b/plugins/connman.c
>> @@ -0,0 +1,277 @@
>> +/*
>> + *
>> + *  oFono - Open Source Telephony
>> + *
>> + *  Copyright (C) 2008-2011  Intel Corporation. All rights reserved.
>> + *
>> + *  This program is free software; you can redistribute it and/or modify
>> + *  it under the terms of the GNU General Public License version 2 as
>> + *  published by the Free Software Foundation.
>> + *
>> + *  This program is distributed in the hope that it will be useful,
>> + *  but WITHOUT ANY WARRANTY; without even the implied warranty of
>> + *  MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
>> + *  GNU General Public License for more details.
>> + *
>> + *  You should have received a copy of the GNU General Public License
>> + *  along with this program; if not, write to the Free Software
>> + *  Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA  02110-1301  USA
>> + *
>> + */
>> +
>> +#ifdef HAVE_CONFIG_H
>> +#include<config.h>
>> +#endif
>> +
>> +#include<errno.h>
>> +#include<stdlib.h>
>> +#include<unistd.h>
>> +
>> +#include<gdbus.h>
>> +#include<string.h>
>> +
>> +#include<ofono.h>
>> +#include<private-network.h>
>> +
> Please get into the habit of running 'make distcheck' before submitting
> patches.  The above header use is actually wrong.
>
> First you should not use ofono.h here, instead you need to include
> <ofono/log.h>,<ofono/dbus.h>  and<ofono/plugin.h>.  ofono.h is strictly
> for private APIs, which you're not using any of in this plugin.
>
> Secondly, private-network.h should be<ofono/private-network.h>  and you
> will need to define OFONO_DBUS_API_SUBJECT_TO_CHANGE.  See just about
> any plugin for an example.
>

Thanks for the info, I didn't know about that.

>> +#define CONNMAN_SERVICE			"net.connman"
>> +#define CONNMAN_PATH			"/net/connman"
>> +
>> +#define CONNMAN_MANAGER_INTERFACE	CONNMAN_SERVICE ".Manager"
>> +#define CONNMAN_MANAGER_PATH		"/"
>> +
>> +static DBusConnection *connection;
>> +static GHashTable *requests;
>> +static unsigned int id;
>> +
>> +struct pns_req {
>> +	int uid;
>> +	DBusPendingCall *pending;
>> +	ofono_private_network_cb_t cb;
>> +	void *data;
>> +	gboolean redundant;
>> +};
>> +
>> +static void pns_release(int uid)
>> +{
>> +	DBusMessage *message = NULL;
>> +	struct pns_req *req;
>> +
>> +	DBG("");
>> +
>> +	req = g_hash_table_lookup(requests,&uid);
>> +	if (!req)
>> +		return;
>> +
>> +	if (req->pending) {
>> +		if (dbus_pending_call_get_completed(req->pending) == FALSE) {
>> +			/*
>> +			 * We want to cancel the request but we have to wait
>> +			 * the response of ConnMan. So we mark request as
>> +			 * redundant until we get the response, then we remove
>> +			 * it from hash table.
>> +			 */
>> +			req->redundant = TRUE;
>> +			return;
>> +		}
>> +	}
> Why do you check both pending and dbus_pending_call_get_completed?  You
> reset pending to NULL pretty early in request_reply.  So at best
> dbus_pending_call_get_completed is redundant, and at worst it will
> actually cause bugs.  Think it through fully.

I will remove this pending reset into request_reply: I have to let dbus 
rountines release the pending call.
I test pending value and dbus_pending_call_get_completed() because it is 
possible that the pending call has been completed and is not NULL.

>> +
>> +	message = dbus_message_new_method_call(CONNMAN_SERVICE,
>> +				     CONNMAN_MANAGER_PATH,
>> +				     CONNMAN_MANAGER_INTERFACE,
>> +				     "ReleasePrivateNetwork");
> You're mixing tabs and spaces for indentation here, please fix this.
> You should also follow the indentation rules of item M4.
>
>> +
>> +	if (message == NULL)
>> +		goto error;
>> +
>> +	dbus_message_set_no_reply(message, TRUE);
>> +	dbus_connection_send(connection, message, NULL);
>> +
> Err, are you missing something important here?  Like the fd argument?
>
> > From connman/doc:
>                  void ReleasePrivateNetwork(fd) [experimental]
>

It seems there is a typo between the doc and the implemented API, for 
the moment ReleasePrivateNetwork() doesn't take any argument.
The PN settings are removed from the hash table using the DBus sender.
I will send a patch to fix the documentation.

>> +error:
>> +	if (message)
>> +		dbus_message_unref(message);
> This check is redundant.  The code would look way nicer if you do:
>
> 	dbus_message_unref(message);
>
> error:
> 	g_hash_table_remove(...);
>
>> +
>> +	g_hash_table_remove(requests,&req->uid);
>> +}
>> +
>> +static void request_reply(DBusPendingCall *call, void *user_data)
>> +{
>> +	struct pns_req *req = user_data;
>> +	struct ofono_private_network_settings pns;
>> +	DBusMessageIter array, dict, entry;
>> +	DBusMessage *reply;
>> +
>> +	DBG("");
>> +
>> +	pns.fd = -1;
>> +	pns.server_ip = NULL;
>> +	pns.peer_ip = NULL;
>> +	pns.primary_dns = NULL;
>> +	pns.secondary_dns = NULL;
>> +
>> +	/* request is no more pending */
>> +	req->pending = NULL;
>> +
>> +	reply = dbus_pending_call_steal_reply(call);
>> +	if (!reply)
>> +		goto error;
>> +
>> +	if (dbus_message_get_type(reply) == DBUS_MESSAGE_TYPE_ERROR)
>> +		goto error;
>> +
>> +	if (dbus_message_iter_init(reply,&array) == FALSE)
>> +		goto error;
>> +
>> +	if (dbus_message_iter_get_arg_type(&array) != DBUS_TYPE_UNIX_FD)
>> +		goto error;
>> +
>> +	dbus_message_iter_get_basic(&array,&pns.fd);
>> +	DBG("Fildescriptor = %d\n", pns.fd);
>> +
>> +	dbus_message_iter_next(&array);
>> +
>> +	if (dbus_message_iter_get_arg_type(&array) != DBUS_TYPE_ARRAY)
>> +		goto error;
>> +
>> +	if (req->redundant == TRUE)
>> +		goto release;
>> +
>> +	dbus_message_iter_recurse(&array,&dict);
>> +
>> +	while (dbus_message_iter_get_arg_type(&dict) == DBUS_TYPE_DICT_ENTRY) {
>> +		DBusMessageIter iter;
>> +		const char *key;
>> +		int type;
>> +
>> +		dbus_message_iter_recurse(&dict,&entry);
>> +
>> +		dbus_message_iter_get_basic(&entry,&key);
>> +
>> +		dbus_message_iter_next(&entry);
>> +		dbus_message_iter_recurse(&entry,&iter);
>> +
>> +		type = dbus_message_iter_get_arg_type(&iter);
>> +		if (type != DBUS_TYPE_STRING)
>> +			break;
>> +
>> +		if (g_str_equal(key, "ServerIPv4")
>> +				&&  type == DBUS_TYPE_STRING)
>> +			dbus_message_iter_get_basic(&iter,&pns.server_ip);
>> +		else if (g_str_equal(key, "PeerIPv4")
>> +				&&  type == DBUS_TYPE_STRING)
>> +			dbus_message_iter_get_basic(&iter,&pns.peer_ip);
>> +		else if (g_str_equal(key, "PrimaryDNS")
>> +				&&  type == DBUS_TYPE_STRING)
>> +			dbus_message_iter_get_basic(&iter,&pns.primary_dns);
>> +		else if (g_str_equal(key, "SecondaryDNS")
>> +				&&  type == DBUS_TYPE_STRING)
>> +			dbus_message_iter_get_basic(&iter,&pns.secondary_dns);
>> +
>> +		dbus_message_iter_next(&dict);
>> +	}
>> +
>> +	if (pns.server_ip == NULL || pns.peer_ip == NULL ||
>> +			pns.primary_dns == NULL || pns.secondary_dns == NULL ||
>> +			pns.fd<  0) {
>> +		ofono_error("Error while reading dictionnary...\n");
>> +		goto release;
>> +	}
>> +
>> +	req->cb(&pns, req->data);
>> +
>> +	dbus_message_unref(reply);
>> +	dbus_pending_call_unref(call);
>> +
>> +	return;
>> +
>> +release:
>> +	pns_release(req->uid);
>> +error:
>> +	if (pns.fd>= 0)
>> +		close(pns.fd);
>> +
>> +	req->cb(NULL, req->data);
>> +
>> +	if (reply)
>> +		dbus_message_unref(reply);
>> +
>> +	dbus_pending_call_unref(call);
>> +}
>> +
>> +static int pns_request(ofono_private_network_cb_t cb, void *data)
>> +{
>> +	DBusMessage *message;
>> +	DBusPendingCall *call;
>> +	struct pns_req *req;
>> +
>> +	DBG("");
>> +
>> +	req = g_try_new(struct pns_req, 1);
>> +
>> +	if (req == NULL)
>> +		return -ENOMEM;
>> +
>> +	message = dbus_message_new_method_call(CONNMAN_SERVICE,
>> +					CONNMAN_MANAGER_PATH,
>> +					CONNMAN_MANAGER_INTERFACE,
>> +					"RequestPrivateNetwork");
> Please align these in accordance with item M4 of our coding-style document.
>
>> +
>> +	if (message == NULL) {
>> +		g_free(req);
>> +		return -ENOMEM;
>> +	}
>> +
>> +	if (dbus_connection_send_with_reply(connection,
>> +				message,&call, 5000) == FALSE) {
> Same here, you can put message on the previous line to make this easier.
>
>> +		g_free(req);
>> +		dbus_message_unref(message);
>> +		return -EIO;
>> +	}
>> +
>> +	id++;
>> +	req->pending = call;
>> +	req->cb = cb;
>> +	req->data = data;
>> +	req->uid = id;
>> +	req->redundant = FALSE;
>> +
>> +	dbus_pending_call_set_notify(call, request_reply,
>> +							req, NULL);
> why are you breaking up this line? It can all fit on the same line.
>
>> +	g_hash_table_insert(requests,&req->uid, req);
>> +	dbus_message_unref(message);
>> +
>> +	return req->uid;
>> +}
>> +
>> +static struct ofono_private_network_driver pn_driver = {
>> +	.name		= "ConnMan Private Network",
>> +	.request	= pns_request,
>> +	.release	= pns_release,
>> +};
>> +
>> +static void remove_requests(gpointer user_data)
>> +{
>> +	struct pns_req *req = user_data;
>> +
>> +	g_free(req);
>> +}
>> +
>> +static int connman_init(void)
>> +{
>> +	DBG("");
>> +
>> +	id = 0;
> The id initialization is redundant.  Either initialize it to zero at
> declaration or just remember that all statics are initialized to zero
> anyway.
>
>> +	connection = ofono_dbus_get_connection();
>> +	requests = g_hash_table_new_full(g_int_hash, g_int_equal, NULL,
>> +						remove_requests);
>> +
>> +	return ofono_private_network_driver_register(&pn_driver);
>> +}
>> +
>> +static void connman_exit(void)
>> +{
>> +	g_hash_table_destroy(requests);
>> +	ofono_private_network_driver_unregister(&pn_driver);
>> +}
>> +
>> +OFONO_PLUGIN_DEFINE(connman, "ConnMan plugin", VERSION,
>> +		OFONO_PLUGIN_PRIORITY_DEFAULT, connman_init, connman_exit)

Kind regards,
Guillaume

      reply	other threads:[~2011-06-08  8:46 UTC|newest]

Thread overview: 3+ messages / expand[flat|nested]  mbox.gz  Atom feed  top
2011-06-06  9:17 [PATCH_v7] connman: add plugin in oFono to request/release private network Guillaume Zajac
2011-06-06  6:17 ` Denis Kenzior
2011-06-08  8:46   ` Guillaume Zajac [this message]

Reply instructions:

You may reply publicly to this message via plain-text email
using any one of the following methods:

* Save the following mbox file, import it into your mail client,
  and reply-to-all from there: mbox

  Avoid top-posting and favor interleaved quoting:
  https://en.wikipedia.org/wiki/Posting_style#Interleaved_style

* Reply using the --to, --cc, and --in-reply-to
  switches of git-send-email(1):

  git send-email \
    --in-reply-to=4DEF36D3.5090106@linux.intel.com \
    --to=guillaume.zajac@linux.intel.com \
    --cc=ofono@ofono.org \
    /path/to/YOUR_REPLY

  https://kernel.org/pub/software/scm/git/docs/git-send-email.html

* If your mail client supports setting the In-Reply-To header
  via mailto: links, try the mailto: link
Be sure your reply has a Subject: header at the top and a blank line before the message body.
This is a public inbox, see mirroring instructions
for how to clone and mirror all data and code used for this inbox