/* Copyright (c) 2014 The Chromium OS Authors. All rights reserved.
 * Use of this source code is governed by a BSD-style license that can be
 * found in the LICENSE file.
 */

#include <string.h>
#include <stdlib.h>
#include <syslog.h>

#include <dbus/dbus.h>

#include "cras_telephony.h"
#include "cras_hfp_ag_profile.h"
#include "cras_hfp_slc.h"

#define CRAS_TELEPHONY_INTERFACE "org.chromium.cras.Telephony"
#define CRAS_TELEPHONY_OBJECT_PATH "/org/chromium/cras/telephony"
#define TELEPHONY_INTROSPECT_XML					\
	DBUS_INTROSPECT_1_0_XML_DOCTYPE_DECL_NODE			\
	"<node>\n"							\
	"  <interface name=\"" CRAS_TELEPHONY_INTERFACE "\">\n"		\
	"    <method name=\"AnswerCall\">\n"				\
	"    </method>\n"						\
	"    <method name=\"IncomingCall\">\n"				\
	"      <arg name=\"value\" type=\"s\" direction=\"in\"/>\n"	\
	"    </method>\n"						\
	"    <method name=\"TerminateCall\">\n"				\
	"    </method>\n"						\
	"    <method name=\"SetBatteryLevel\">\n"				\
	"      <arg name=\"value\" type=\"i\" direction=\"in\"/>\n"	\
	"    </method>\n"						\
	"    <method name=\"SetSignalStrength\">\n"				\
	"      <arg name=\"value\" type=\"i\" direction=\"in\"/>\n"	\
	"    </method>\n"						\
	"    <method name=\"SetServiceAvailability\">\n"				\
	"      <arg name=\"value\" type=\"i\" direction=\"in\"/>\n"	\
	"    </method>\n"						\
	"    <method name=\"SetDialNumber\">\n"				\
	"      <arg name=\"value\" type=\"s\" direction=\"in\"/>\n"	\
	"    </method>\n"						\
	"    <method name=\"SetCallheld\">\n"				\
	"      <arg name=\"value\" type=\"i\" direction=\"in\"/>\n"	\
	"    </method>\n"						\
	"  </interface>\n"						\
	"  <interface name=\"" DBUS_INTERFACE_INTROSPECTABLE "\">\n"	\
	"    <method name=\"Introspect\">\n"				\
	"      <arg name=\"data\" type=\"s\" direction=\"out\"/>\n"	\
	"    </method>\n"						\
	"  </interface>\n"						\
	"</node>\n"

static struct cras_telephony_handle telephony_handle;

/* Helper to extract a single argument from a DBus message. */
static DBusHandlerResult get_single_arg(DBusMessage *message,
					int dbus_type, void *arg)
{
	DBusError dbus_error;

	dbus_error_init(&dbus_error);

	if (!dbus_message_get_args(message, &dbus_error,
				   dbus_type, arg,
				   DBUS_TYPE_INVALID)) {
		syslog(LOG_WARNING,
		       "Bad method received: %s",
		       dbus_error.message);
		dbus_error_free(&dbus_error);
		return DBUS_HANDLER_RESULT_NOT_YET_HANDLED;
	}

	return DBUS_HANDLER_RESULT_HANDLED;
}

/* Helper to send an empty reply. */
static void send_empty_reply(DBusConnection *conn, DBusMessage *message)
{
	DBusMessage *reply;
	dbus_uint32_t serial = 0;

	reply = dbus_message_new_method_return(message);
	if (!reply)
		return;

	dbus_connection_send(conn, reply, &serial);

	dbus_message_unref(reply);
}

static DBusHandlerResult handle_incoming_call(DBusConnection *conn,
					      DBusMessage *message,
					      void *arg)
{
	struct hfp_slc_handle *handle;
	DBusHandlerResult rc;
	const char* number;

	rc = get_single_arg(message, DBUS_TYPE_STRING, &number);
	if (rc != DBUS_HANDLER_RESULT_HANDLED)
		return rc;

	handle = cras_hfp_ag_get_active_handle();

	telephony_handle.callsetup = 1;

	if (handle) {
		hfp_event_update_callsetup(handle);
		hfp_event_incoming_call(handle, number, 129);
	}

	send_empty_reply(conn, message);
	return DBUS_HANDLER_RESULT_HANDLED;
}

static DBusHandlerResult handle_terminate_call(DBusConnection *conn,
					       DBusMessage *message,
					       void *arg)
{
	cras_telephony_event_terminate_call();

	send_empty_reply(conn, message);
	return DBUS_HANDLER_RESULT_HANDLED;
}

static DBusHandlerResult handle_answer_call(DBusConnection *conn,
					    DBusMessage *message,
					    void *arg)
{
	cras_telephony_event_answer_call();

	send_empty_reply(conn, message);
	return DBUS_HANDLER_RESULT_HANDLED;
}

static DBusHandlerResult handle_set_dial_number(DBusConnection *conn,
						DBusMessage *message,
						void *arg)
{
	DBusHandlerResult rc;
	const char *number;

	rc = get_single_arg(message, DBUS_TYPE_STRING, &number);
	if (rc != DBUS_HANDLER_RESULT_HANDLED)
		return rc;

	cras_telephony_store_dial_number(strlen(number), number);

	send_empty_reply(conn, message);
	return DBUS_HANDLER_RESULT_HANDLED;
}

static DBusHandlerResult handle_set_battery(DBusConnection *conn,
					    DBusMessage *message,
					    void *arg)
{
	struct hfp_slc_handle *handle;
	DBusHandlerResult rc;
	int value;

	rc = get_single_arg(message, DBUS_TYPE_INT32, &value);
	if (rc != DBUS_HANDLER_RESULT_HANDLED)
		return rc;

	handle = cras_hfp_ag_get_active_handle();
	if (handle)
		hfp_event_set_battery(handle, value);

	send_empty_reply(conn, message);
	return DBUS_HANDLER_RESULT_HANDLED;
}

static DBusHandlerResult handle_set_signal(DBusConnection *conn,
					   DBusMessage *message,
					   void *arg)
{
	struct hfp_slc_handle *handle;
	DBusHandlerResult rc;
	int value;

	rc = get_single_arg(message, DBUS_TYPE_INT32, &value);
	if (rc != DBUS_HANDLER_RESULT_HANDLED)
		return rc;

	handle = cras_hfp_ag_get_active_handle();
	if (handle)
		hfp_event_set_signal(handle, value);

	send_empty_reply(conn, message);
	return DBUS_HANDLER_RESULT_HANDLED;
}

static DBusHandlerResult handle_set_service(DBusConnection *conn,
					    DBusMessage *message,
					    void *arg)
{
	struct hfp_slc_handle *handle;
	DBusHandlerResult rc;
	int value;

	rc = get_single_arg(message, DBUS_TYPE_INT32, &value);
	if (rc != DBUS_HANDLER_RESULT_HANDLED)
		return rc;

	handle = cras_hfp_ag_get_active_handle();
	if (handle)
		hfp_event_set_service(handle, value);

	send_empty_reply(conn, message);
	return DBUS_HANDLER_RESULT_HANDLED;
}

static DBusHandlerResult handle_set_callheld(DBusConnection *conn,
					     DBusMessage *message,
					     void *arg)
{
	struct hfp_slc_handle *handle;
	DBusHandlerResult rc;
	int value;

	rc = get_single_arg(message, DBUS_TYPE_INT32, &value);
	if (rc != DBUS_HANDLER_RESULT_HANDLED)
		return rc;

	telephony_handle.callheld = value;
	handle = cras_hfp_ag_get_active_handle();
	if (handle)
		hfp_event_update_callheld(handle);

	send_empty_reply(conn, message);
	return DBUS_HANDLER_RESULT_HANDLED;
}

/* Handle incoming messages. */
static DBusHandlerResult handle_telephony_message(DBusConnection *conn,
						  DBusMessage *message,
						  void *arg)
{
	syslog(LOG_ERR, "Telephony message: %s %s %s",
			dbus_message_get_path(message),
			dbus_message_get_interface(message),
			dbus_message_get_member(message));

	if (dbus_message_is_method_call(message,
					DBUS_INTERFACE_INTROSPECTABLE,
					"Introspect")) {
		DBusMessage *reply;
		const char *xml = TELEPHONY_INTROSPECT_XML;

		reply = dbus_message_new_method_return(message);
		if (!reply)
			return DBUS_HANDLER_RESULT_NEED_MEMORY;
		if (!dbus_message_append_args(reply,
					      DBUS_TYPE_STRING, &xml,
					      DBUS_TYPE_INVALID))
			return DBUS_HANDLER_RESULT_NEED_MEMORY;
		if (!dbus_connection_send(conn, reply, NULL))
			return DBUS_HANDLER_RESULT_NEED_MEMORY;

		dbus_message_unref(reply);
		return DBUS_HANDLER_RESULT_HANDLED;
	} else if (dbus_message_is_method_call(message,
					       CRAS_TELEPHONY_INTERFACE,
					       "IncomingCall")) {
		return handle_incoming_call(conn, message, arg);
	} else if (dbus_message_is_method_call(message,
					       CRAS_TELEPHONY_INTERFACE,
					       "TerminateCall")) {
		return handle_terminate_call(conn, message, arg);
	} else if (dbus_message_is_method_call(message,
					       CRAS_TELEPHONY_INTERFACE,
					       "AnswerCall")) {
		return handle_answer_call(conn, message, arg);
	} else if (dbus_message_is_method_call(message,
					       CRAS_TELEPHONY_INTERFACE,
					       "SetDialNumber")) {
		return handle_set_dial_number(conn, message, arg);
	} else if (dbus_message_is_method_call(message,
					       CRAS_TELEPHONY_INTERFACE,
					       "SetBatteryLevel")) {
		return handle_set_battery(conn, message, arg);
	} else if (dbus_message_is_method_call(message,
					       CRAS_TELEPHONY_INTERFACE,
					       "SetSignalStrength")) {
		return handle_set_signal(conn, message, arg);
	} else if (dbus_message_is_method_call(message,
					       CRAS_TELEPHONY_INTERFACE,
					       "SetServiceAvailability")) {
		return handle_set_service(conn, message, arg);
	} else if (dbus_message_is_method_call(message,
					       CRAS_TELEPHONY_INTERFACE,
					       "SetCallheld")) {
		return handle_set_callheld(conn, message, arg);
	}

	return DBUS_HANDLER_RESULT_NOT_YET_HANDLED;
}

/* Exported Interface */

void cras_telephony_start(DBusConnection *conn)
{
	static const DBusObjectPathVTable control_vtable = {
		.message_function = handle_telephony_message,
	};

	DBusError dbus_error;

	telephony_handle.dbus_conn = conn;
	dbus_connection_ref(telephony_handle.dbus_conn);

	if (!dbus_connection_register_object_path(conn,
						  CRAS_TELEPHONY_OBJECT_PATH,
						  &control_vtable,
						  &dbus_error)) {
		syslog(LOG_ERR,
		       "Couldn't register telephony control: %s: %s",
		       CRAS_TELEPHONY_OBJECT_PATH, dbus_error.message);
		dbus_error_free(&dbus_error);
		return;
	}
}

void cras_telephony_stop()
{
	if (!telephony_handle.dbus_conn)
		return;

	dbus_connection_unregister_object_path(telephony_handle.dbus_conn,
					       CRAS_TELEPHONY_OBJECT_PATH);
	dbus_connection_unref(telephony_handle.dbus_conn);
	telephony_handle.dbus_conn = NULL;
}

struct cras_telephony_handle* cras_telephony_get()
{
	return &telephony_handle;
}

/* Procedure to answer a call from AG.
 *
 * HF(hands-free)                             AG(audio gateway)
 *                                                     <-- Call answered
 *                 <-- +CIEV: (call = 1)
 *                 <-- +CIEV: (callsetup = 0)
 */
int cras_telephony_event_answer_call()
{
	int rc;

	struct hfp_slc_handle *handle;

	handle = cras_hfp_ag_get_active_handle();

	if (telephony_handle.call == 0) {
		telephony_handle.call = 1;
		if (handle) {
			rc = hfp_event_update_call(handle);
			if (rc)
				return rc;
		}
	}

	telephony_handle.callsetup = 0;
	if (handle) {
		rc = hfp_event_update_callsetup(handle);
		if (rc)
			return rc;
	}

	return 0;
}

/* Procedure to terminate a call from AG.
 *
 * HF(hands-free)                             AG(audio gateway)
 *                                                     <-- Call dropped
 *                 <-- +CIEV: (call = 0)
 */
int cras_telephony_event_terminate_call()
{
	int rc;
	struct hfp_slc_handle *handle;

	handle = cras_hfp_ag_get_active_handle();

	if (telephony_handle.call) {
		telephony_handle.call = 0;
		if (handle) {
			rc = hfp_event_update_call(handle);
			if (rc)
				return rc;
		}
	}
	if (telephony_handle.callsetup) {
		telephony_handle.callsetup = 0;
		if (handle) {
			rc = hfp_event_update_callsetup(handle);
			if (rc)
				return rc;
		}
	}
	return 0;
}

void cras_telephony_store_dial_number(int len,
				      const char *number)
{
	if (telephony_handle.dial_number != NULL) {
		free(telephony_handle.dial_number);
		telephony_handle.dial_number = NULL;
	}

	if (len == 0)
		return ;

	telephony_handle.dial_number =
			(char *) calloc(len + 1,
					sizeof(*telephony_handle.dial_number));
	strncpy(telephony_handle.dial_number, number, len);

	syslog(LOG_ERR,
	       "store dial_number: \"%s\"", telephony_handle.dial_number);
}