| 1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
 | #include <stdlib.h>
#include <stdbool.h>
#include <stdint.h>
#include "tapi-device.h"
#include "tapi-port.h"
#include "tapi-session.h"
enum tapi_session_port_state {
	TAPI_SESSION_PORT_STATE_IDLE,
	TAPI_SESSION_PORT_STATE_RINGING,
	TAPI_SESSION_PORT_STATE_ACTIVE,
};
struct tapi_session_port {
	struct tapi_port *port;
	struct tapi_port_event_listener event_listener;
	enum tapi_session_port_state state;
};
struct tapi_session {
	struct tapi_device *dev;
	struct tapi_session_port caller;
	struct tapi_session_port callee;
	bool active;
	unsigned int link;
	void (*release)(struct tapi_session *session, void *data);
};
static void tapi_session_terminate(struct tapi_session *session)
{
	if (session->active) {
		tapi_link_enable(session->dev, session->link);
		tapi_sync(session->dev);
		tapi_link_free(session->dev, session->link);
	}
	switch (session->callee.state) {
	case TAPI_SESSION_PORT_STATE_RINGING:
		tapi_port_set_ring(session->callee.port, false);
		break;
	default:
		break;
	}
	session->active = false;
}
static void tapi_session_caller_event(struct tapi_port *port,
	struct tapi_event *event, void *data)
{
	struct tapi_session *session = data;
	if (event->type != TAPI_EVENT_TYPE_HOOK)
		return;
	if (event->hook.on) {
		tapi_session_terminate(session);
	}
}
static void tapi_session_callee_event(struct tapi_port *port,
	struct tapi_event *event, void *data)
{
	struct tapi_session *session = data;
	if (event->type != TAPI_EVENT_TYPE_HOOK)
		return;
	if (event->hook.on) {
		if (session->callee.state == TAPI_SESSION_PORT_STATE_ACTIVE) {
			tapi_session_terminate(session);
		}
	} else  {
		if (session->callee.state == TAPI_SESSION_PORT_STATE_RINGING) {
			tapi_port_set_ring(session->callee.port, false);
			session->link = tapi_link_alloc(session->dev,
				session->caller.port->ep, session->callee.port->ep);
			session->callee.state = TAPI_SESSION_PORT_STATE_ACTIVE;
			tapi_link_enable(session->dev, session->link);
			tapi_sync(session->dev);
			session->active = true;
		}
	}
}
struct tapi_session *tapi_session_alloc(struct tapi_device *dev,
	struct tapi_port *caller, struct tapi_port *callee,
	void (*release)(struct tapi_session *session, void *data), void *release_data)
{
	struct tapi_session *session;
	struct tapi_session_port *session_port;
	session = malloc(sizeof(*session));
	session->dev = dev;
	session->callee.port = callee;
	session->callee.state = TAPI_SESSION_PORT_STATE_RINGING;
	session->callee.event_listener.callback = tapi_session_callee_event;
	session->callee.event_listener.data = session;
	tapi_port_register_event(callee, &session->callee.event_listener);
	session->caller.port = caller;
	session->caller.state = TAPI_SESSION_PORT_STATE_ACTIVE;
	session->caller.event_listener.callback = tapi_session_caller_event;
	session->caller.event_listener.data = session;
	tapi_port_register_event(caller, &session->caller.event_listener);
	tapi_port_set_ring(callee, true);
}
void tapi_session_free(struct tapi_session *session)
{
	tapi_session_terminate(session);
	tapi_port_register_event(session->callee.port, &session->callee.event_listener);
	tapi_port_register_event(session->caller.port, &session->caller.event_listener);
	free(session);
}
 |