mirror of
https://gitlab.gnome.org/GNOME/calls.git
synced 2025-01-08 12:55:32 +00:00
b1a743b757
This actually makes sense in the mapping; the only time an MM call is in MM_CALL_STATE_UNKNOWN is when it's an outgoing call that hasn't been started. It also helps avoid a lot of needless issues dealing with the Calls call state.
463 lines
12 KiB
C
463 lines
12 KiB
C
/*
|
|
* Copyright (C) 2018 Purism SPC
|
|
*
|
|
* This file is part of Calls.
|
|
*
|
|
* Calls is free software: you can redistribute it and/or modify it
|
|
* under the terms of the GNU General Public License as published by
|
|
* the Free Software Foundation, either version 3 of the License, or
|
|
* (at your option) any later version.
|
|
*
|
|
* Calls 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 Calls. If not, see <http://www.gnu.org/licenses/>.
|
|
*
|
|
* Author: Bob Ham <bob.ham@puri.sm>
|
|
*
|
|
* SPDX-License-Identifier: GPL-3.0-or-later
|
|
*
|
|
*/
|
|
|
|
#include "calls-mm-call.h"
|
|
#include "calls-call.h"
|
|
#include "calls-message-source.h"
|
|
#include "util.h"
|
|
|
|
#include <libmm-glib/libmm-glib.h>
|
|
#include <glib/gi18n.h>
|
|
|
|
|
|
struct _CallsMMCall
|
|
{
|
|
GObject parent_instance;
|
|
MMCall *mm_call;
|
|
GString *number;
|
|
CallsCallState state;
|
|
gchar *disconnect_reason;
|
|
};
|
|
|
|
static void calls_mm_call_message_source_interface_init (CallsCallInterface *iface);
|
|
static void calls_mm_call_call_interface_init (CallsCallInterface *iface);
|
|
|
|
G_DEFINE_TYPE_WITH_CODE (CallsMMCall, calls_mm_call, G_TYPE_OBJECT,
|
|
G_IMPLEMENT_INTERFACE (CALLS_TYPE_MESSAGE_SOURCE,
|
|
calls_mm_call_message_source_interface_init)
|
|
G_IMPLEMENT_INTERFACE (CALLS_TYPE_CALL,
|
|
calls_mm_call_call_interface_init))
|
|
|
|
enum {
|
|
PROP_0,
|
|
PROP_MM_CALL,
|
|
PROP_INBOUND,
|
|
PROP_LAST_PROP,
|
|
};
|
|
static GParamSpec *props[PROP_LAST_PROP];
|
|
|
|
|
|
static const gchar *
|
|
get_name (CallsCall *call)
|
|
{
|
|
return NULL;
|
|
}
|
|
|
|
|
|
static const gchar *
|
|
get_number (CallsCall *call)
|
|
{
|
|
CallsMMCall *self = CALLS_MM_CALL (call);
|
|
return self->number->str;
|
|
}
|
|
|
|
|
|
static CallsCallState
|
|
get_state (CallsCall *call)
|
|
{
|
|
CallsMMCall *self = CALLS_MM_CALL (call);
|
|
return self->state;
|
|
}
|
|
|
|
|
|
static void
|
|
change_state (CallsMMCall *self,
|
|
CallsCallState state)
|
|
{
|
|
CallsCallState old_state = self->state;
|
|
|
|
if (old_state == state)
|
|
{
|
|
return;
|
|
}
|
|
|
|
self->state = state;
|
|
g_signal_emit_by_name (CALLS_CALL (self),
|
|
"state-changed",
|
|
state,
|
|
old_state);
|
|
}
|
|
|
|
|
|
static void
|
|
notify_number_cb (CallsMMCall *self,
|
|
const gchar *number)
|
|
{
|
|
g_string_assign (self->number, number);
|
|
}
|
|
|
|
|
|
struct CallsMMCallStateReasonMap
|
|
{
|
|
MMCallStateReason value;
|
|
const gchar *desc;
|
|
};
|
|
|
|
static const struct CallsMMCallStateReasonMap STATE_REASON_MAP[] = {
|
|
|
|
#define row(ENUMVALUE,DESCRIPTION) \
|
|
{ MM_CALL_STATE_REASON_##ENUMVALUE, DESCRIPTION } \
|
|
|
|
row (UNKNOWN, "Outgoing call created"),
|
|
row (OUTGOING_STARTED, "Outgoing call started"),
|
|
row (INCOMING_NEW, "Incoming call"),
|
|
row (ACCEPTED, "Call accepted"),
|
|
row (TERMINATED, "Call terminated"),
|
|
row (REFUSED_OR_BUSY, "Busy or call refused"),
|
|
row (ERROR, "Wrong number or network problem"),
|
|
|
|
#undef row
|
|
|
|
{ -1, NULL }
|
|
};
|
|
|
|
static void
|
|
set_disconnect_reason (CallsMMCall *self,
|
|
MMCallStateReason reason)
|
|
{
|
|
const struct CallsMMCallStateReasonMap *map_row;
|
|
GString *reason_str;
|
|
|
|
for (map_row = STATE_REASON_MAP; map_row->desc; ++map_row)
|
|
{
|
|
if (map_row->value == reason)
|
|
{
|
|
self->disconnect_reason = g_strdup(map_row->desc);
|
|
return;
|
|
}
|
|
}
|
|
|
|
reason_str = g_string_new ("Unknown disconnect reason ");
|
|
g_string_append_printf (reason_str, "(%i)", (int)reason);
|
|
|
|
g_warning ("%s", reason_str->str);
|
|
CALLS_SET_PTR_PROPERTY (self->disconnect_reason,
|
|
reason_str->str);
|
|
|
|
g_string_free (reason_str, FALSE);
|
|
}
|
|
|
|
|
|
struct CallsMMCallStateMap
|
|
{
|
|
MMCallState mm;
|
|
CallsCallState calls;
|
|
const gchar *name;
|
|
};
|
|
|
|
static const struct CallsMMCallStateMap STATE_MAP[] = {
|
|
|
|
#define row(MMENUM,CALLSENUM) \
|
|
{ MM_CALL_STATE_##MMENUM, CALLS_CALL_STATE_##CALLSENUM, #MMENUM } \
|
|
|
|
row (UNKNOWN, DIALING),
|
|
row (DIALING, DIALING),
|
|
row (RINGING_OUT, ALERTING),
|
|
row (RINGING_IN, INCOMING),
|
|
row (ACTIVE, ACTIVE),
|
|
row (HELD, HELD),
|
|
row (WAITING, INCOMING),
|
|
row (TERMINATED, DISCONNECTED),
|
|
|
|
#undef row
|
|
|
|
{ -1, -1 }
|
|
};
|
|
|
|
static void
|
|
state_changed_cb (CallsMMCall *self,
|
|
MMCallState old,
|
|
MMCallState mm_new,
|
|
MMCallStateReason reason)
|
|
{
|
|
const struct CallsMMCallStateMap *map_row;
|
|
|
|
if (mm_new == MM_CALL_STATE_TERMINATED)
|
|
{
|
|
set_disconnect_reason (self, reason);
|
|
}
|
|
|
|
for (map_row = STATE_MAP; map_row->mm != -1; ++map_row)
|
|
{
|
|
if (map_row->mm == mm_new)
|
|
{
|
|
g_debug ("MM call state changed to `%s'",
|
|
map_row->name);
|
|
change_state (self, map_row->calls);
|
|
return;
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
struct CallsMMOperationData
|
|
{
|
|
const gchar *desc;
|
|
CallsMMCall *self;
|
|
gboolean (*finish_func) (MMCall *, GAsyncResult *, GError **);
|
|
};
|
|
|
|
static void
|
|
operation_cb (MMCall *mm_call,
|
|
GAsyncResult *res,
|
|
struct CallsMMOperationData *data)
|
|
{
|
|
gboolean ok;
|
|
GError *error = NULL;
|
|
|
|
ok = data->finish_func (mm_call, res, &error);
|
|
if (!ok)
|
|
{
|
|
g_warning ("Error %s ModemManager call to `%s': %s",
|
|
data->desc, data->self->number->str, error->message);
|
|
CALLS_ERROR (data->self, error);
|
|
}
|
|
|
|
g_free (data);
|
|
}
|
|
|
|
#define DEFINE_OPERATION(op,name,desc_str) \
|
|
static void \
|
|
name (CallsCall *call) \
|
|
{ \
|
|
CallsMMCall *self = CALLS_MM_CALL (call); \
|
|
struct CallsMMOperationData *data; \
|
|
\
|
|
data = g_new0 (struct CallsMMOperationData, 1); \
|
|
data->desc = desc_str; \
|
|
data->self = self; \
|
|
data->finish_func = mm_call_##op##_finish; \
|
|
\
|
|
mm_call_##op \
|
|
(self->mm_call, \
|
|
NULL, \
|
|
(GAsyncReadyCallback) operation_cb, \
|
|
data); \
|
|
}
|
|
|
|
DEFINE_OPERATION(accept, answer, "accepting");
|
|
DEFINE_OPERATION(hangup, hang_up, "hanging up");
|
|
DEFINE_OPERATION(start, start_call, "starting outgoing call");
|
|
|
|
|
|
static void
|
|
tone_start (CallsCall *call, gchar key)
|
|
{
|
|
CallsMMCall *self = CALLS_MM_CALL (call);
|
|
struct CallsMMOperationData *data;
|
|
char key_str[2] = { key, '\0' };
|
|
|
|
data = g_new0 (struct CallsMMOperationData, 1);
|
|
data->desc = "sending DTMF";
|
|
data->self = self;
|
|
data->finish_func = mm_call_send_dtmf_finish;
|
|
|
|
mm_call_send_dtmf
|
|
(self->mm_call,
|
|
key_str,
|
|
NULL,
|
|
(GAsyncReadyCallback) operation_cb,
|
|
data);
|
|
}
|
|
|
|
|
|
static void
|
|
set_property (GObject *object,
|
|
guint property_id,
|
|
const GValue *value,
|
|
GParamSpec *pspec)
|
|
{
|
|
CallsMMCall *self = CALLS_MM_CALL (object);
|
|
|
|
switch (property_id) {
|
|
case PROP_MM_CALL:
|
|
g_set_object (&self->mm_call,
|
|
MM_CALL (g_value_get_object (value)));
|
|
break;
|
|
|
|
default:
|
|
G_OBJECT_WARN_INVALID_PROPERTY_ID (object, property_id, pspec);
|
|
break;
|
|
}
|
|
}
|
|
|
|
|
|
static void
|
|
constructed (GObject *object)
|
|
{
|
|
GObjectClass *parent_class = g_type_class_peek (G_TYPE_OBJECT);
|
|
CallsMMCall *self = CALLS_MM_CALL (object);
|
|
MmGdbusCall *gdbus_call = MM_GDBUS_CALL (self->mm_call);
|
|
MMCallState state;
|
|
|
|
g_signal_connect_swapped (gdbus_call, "notify::number",
|
|
G_CALLBACK (notify_number_cb), self);
|
|
g_signal_connect_swapped (gdbus_call, "state-changed",
|
|
G_CALLBACK (state_changed_cb), self);
|
|
|
|
notify_number_cb (self, mm_call_get_number (self->mm_call));
|
|
|
|
state = mm_call_get_state (self->mm_call);
|
|
state_changed_cb (self,
|
|
MM_MODEM_STATE_UNKNOWN,
|
|
state,
|
|
mm_call_get_state_reason (self->mm_call));
|
|
|
|
/* Start outgoing call */
|
|
if (state == MM_CALL_STATE_UNKNOWN
|
|
&& mm_call_get_direction (self->mm_call) == MM_CALL_DIRECTION_OUTGOING)
|
|
{
|
|
start_call (CALLS_CALL (self));
|
|
}
|
|
|
|
parent_class->constructed (object);
|
|
}
|
|
|
|
|
|
static void
|
|
get_property (GObject *object,
|
|
guint property_id,
|
|
GValue *value,
|
|
GParamSpec *pspec)
|
|
{
|
|
CallsMMCall *self = CALLS_MM_CALL (object);
|
|
|
|
switch (property_id) {
|
|
case PROP_INBOUND:
|
|
g_value_set_boolean (value,
|
|
mm_call_get_direction (self->mm_call)
|
|
== MM_CALL_DIRECTION_INCOMING);
|
|
break;
|
|
|
|
default:
|
|
G_OBJECT_WARN_INVALID_PROPERTY_ID (object, property_id, pspec);
|
|
break;
|
|
}
|
|
}
|
|
|
|
|
|
static void
|
|
dispose (GObject *object)
|
|
{
|
|
GObjectClass *parent_class = g_type_class_peek (G_TYPE_OBJECT);
|
|
CallsMMCall *self = CALLS_MM_CALL (object);
|
|
|
|
g_clear_object (&self->mm_call);
|
|
|
|
parent_class->dispose (object);
|
|
}
|
|
|
|
|
|
static void
|
|
finalize (GObject *object)
|
|
{
|
|
GObjectClass *parent_class = g_type_class_peek (G_TYPE_OBJECT);
|
|
CallsMMCall *self = CALLS_MM_CALL (object);
|
|
|
|
g_free (self->disconnect_reason);
|
|
g_string_free (self->number, TRUE);
|
|
|
|
parent_class->finalize (object);
|
|
}
|
|
|
|
|
|
static void
|
|
calls_mm_call_class_init (CallsMMCallClass *klass)
|
|
{
|
|
GObjectClass *object_class = G_OBJECT_CLASS (klass);
|
|
|
|
object_class->get_property = get_property;
|
|
object_class->set_property = set_property;
|
|
object_class->constructed = constructed;
|
|
object_class->dispose = dispose;
|
|
object_class->finalize = finalize;
|
|
|
|
props[PROP_MM_CALL] =
|
|
g_param_spec_object ("mm-call",
|
|
_("MM call"),
|
|
_("A libmm-glib proxy object for the underlying call object"),
|
|
MM_TYPE_CALL,
|
|
G_PARAM_WRITABLE | G_PARAM_CONSTRUCT_ONLY);
|
|
g_object_class_install_property (object_class, PROP_MM_CALL,
|
|
props[PROP_MM_CALL]);
|
|
|
|
|
|
props[PROP_INBOUND] =
|
|
g_param_spec_boolean ("inbound",
|
|
_("Inbound"),
|
|
_("Whether the call is inbound"),
|
|
FALSE,
|
|
G_PARAM_READABLE | G_PARAM_CONSTRUCT);
|
|
g_object_class_override_property (object_class, PROP_INBOUND, "inbound");
|
|
}
|
|
|
|
|
|
static void
|
|
calls_mm_call_message_source_interface_init (CallsCallInterface *iface)
|
|
{
|
|
}
|
|
|
|
|
|
static void
|
|
calls_mm_call_call_interface_init (CallsCallInterface *iface)
|
|
{
|
|
iface->get_number = get_number;
|
|
iface->get_name = get_name;
|
|
iface->get_state = get_state;
|
|
iface->answer = answer;
|
|
iface->hang_up = hang_up;
|
|
iface->tone_start = tone_start;
|
|
}
|
|
|
|
|
|
static void
|
|
calls_mm_call_init (CallsMMCall *self)
|
|
{
|
|
self->number = g_string_new (NULL);
|
|
}
|
|
|
|
|
|
CallsMMCall *
|
|
calls_mm_call_new (MMCall *mm_call)
|
|
{
|
|
g_return_val_if_fail (MM_IS_CALL (mm_call), NULL);
|
|
|
|
return g_object_new (CALLS_TYPE_MM_CALL,
|
|
"mm-call", mm_call,
|
|
NULL);
|
|
}
|
|
|
|
|
|
const gchar *
|
|
calls_mm_call_get_object_path (CallsMMCall *call)
|
|
{
|
|
return mm_call_get_path (call->mm_call);
|
|
}
|
|
|
|
|
|
const gchar *
|
|
calls_mm_call_get_disconnect_reason (CallsMMCall *call)
|
|
{
|
|
return call->disconnect_reason;
|
|
}
|