summaryrefslogtreecommitdiff
path: root/examples/echo/echo-client.cpp
blob: f01f23a9a58028eba4e6cae7ff5637948770bfaa (plain)
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
123
124
125
126
127
128
129
130
131
132
133
134
#ifdef HAVE_CONFIG_H
#include <config.h>
#endif

#include "echo-client.h"
#include <iostream>
#include <pthread.h>
#include <signal.h>
#include <stdio.h>
#include <cstring>

using namespace std;

static const char *ECHO_SERVER_NAME = "org.freedesktop.DBus.Examples.Echo";
static const char *ECHO_SERVER_PATH = "/org/freedesktop/DBus/Examples/Echo";

EchoClient::EchoClient(DBus::Connection &connection, const char *path, const char *name)
: DBus::ObjectProxy(connection, path, name)
{
}

void EchoClient::Echoed(const DBus::Variant &value)
{
	cout << "!";
}

/*
 * For some strange reason, libdbus frequently dies with an OOM
 */

static const int THREADS = 3;

static bool spin = true;

EchoClient *g_client = NULL;

DBus::Pipe *thread_pipe_list[THREADS];

DBus::BusDispatcher dispatcher;
DBus::DefaultTimeout *timeout;

void *greeter_thread(void *arg)
{
	char idstr[16];
	int i = (int) arg;

	snprintf(idstr, sizeof(idstr), "%lu", pthread_self());

	thread_pipe_list[i]->write (idstr, strlen (idstr) + 1);

	cout << idstr << " done (" << i << ")" << endl;

	return NULL;
}

void niam(int sig)
{
	spin = false;

	dispatcher.leave();
}

void handler1 (const void *data, void *buffer, unsigned int nbyte)
{
	char *str = (char*) buffer;
	cout << "buffer1: " << str << ", size: " << nbyte << endl;
	for (int i = 0; i < 30 && spin; ++i)
	{
		cout << "call1: " << g_client->Hello (str) << endl;
	}
}

void handler2 (const void *data, void *buffer, unsigned int nbyte)
{
	char *str = (char*) buffer;
	cout << "buffer2: " << str << ", size: " << nbyte <<endl;
	for (int i = 0; i < 30 && spin; ++i)
	{
		cout << "call2: " << g_client->Hello (str) << endl;
	}
}

void handler3 (const void *data, void *buffer, unsigned int nbyte)
{
	char *str = (char*) buffer;
	cout << "buffer3: " << str << ", size: " << nbyte <<endl;
	for (int i = 0; i < 30 && spin; ++i)
	{
		cout << "call3: " << g_client->Hello (str) << endl;
	}
}

int main()
{
	signal(SIGTERM, niam);
	signal(SIGINT, niam);

	DBus::_init_threading();

  DBus::default_dispatcher = &dispatcher;

  // increase DBus-C++ frequency
  new DBus::DefaultTimeout(100, false, &dispatcher);

	DBus::Connection conn = DBus::Connection::SessionBus();

	EchoClient client (conn, ECHO_SERVER_PATH, ECHO_SERVER_NAME);
	g_client = &client;

	pthread_t threads[THREADS];

	thread_pipe_list[0] = dispatcher.add_pipe (handler1, NULL);
	thread_pipe_list[1] = dispatcher.add_pipe (handler2, NULL);
	thread_pipe_list[2] = dispatcher.add_pipe (handler3, NULL);
	for (int i = 0; i < THREADS; ++i)
	{
		pthread_create(threads+i, NULL, greeter_thread, (void*) i);
	}
	
	dispatcher.enter();

	cout << "terminating" << endl;

	for (int i = 0; i < THREADS; ++i)
	{
		pthread_join(threads[i], NULL);
	}

	dispatcher.del_pipe (thread_pipe_list[0]);
	dispatcher.del_pipe (thread_pipe_list[1]);
	dispatcher.del_pipe (thread_pipe_list[2]);

	return 0;
}