2 * Copyright (c) 2012 Samsung Electronics Co., Ltd All Rights Reserved
4 * Licensed under the Apache License, Version 2.0 (the "License");
5 * you may not use this file except in compliance with the License.
6 * You may obtain a copy of the License at
8 * http://www.apache.org/licenses/LICENSE-2.0
10 * Unless required by applicable law or agreed to in writing, software
11 * distributed under the License is distributed on an "AS IS" BASIS,
12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13 * See the License for the specific language governing permissions and
14 * limitations under the License.
17 /* standard library header */
21 #include <glib-object.h>
25 #include "SEService.h"
26 #include "ClientChannel.h"
28 #include "ClientGDBus.h"
31 #define EXTERN_API __attribute__((visibility("default")))
34 #define SHUTDOWN_DELAY 500000 /* us */
37 namespace smartcard_service_api
39 SEService::SEService() : SEServiceHelper(),
40 handle(-1), context(NULL), handler(NULL), listener(NULL),
46 SEService::SEService(void *user_data, serviceConnected handler)
47 throw(ErrorIO &, ErrorIllegalParameter &) :
48 SEServiceHelper(), handle(-1),
49 listener(NULL), version(VERSION)
51 initialize(user_data, handler);
54 SEService::SEService(void *user_data, SEServiceListener *listener)
55 throw(ErrorIO &, ErrorIllegalParameter &) :
56 SEServiceHelper(), handle(-1),
57 handler(NULL), version(VERSION)
59 initialize(user_data, listener);
62 SEService::SEService(void *user_data)
63 throw(ErrorIO &, ErrorIllegalParameter &, ExceptionBase &) :
64 SEServiceHelper(), handle(-1),
65 handler(NULL), version(VERSION)
67 initializeSync(user_data);
70 SEService::~SEService()
78 for (i = 0; i < readers.size(); i++)
80 delete (Reader *)readers[i];
85 catch (ExceptionBase &e)
87 _ERR("EXCEPTION : %s", e.what());
95 SEService *SEService::createInstance(void *user_data,
96 SEServiceListener *listener)
97 throw(ErrorIO &, ErrorIllegalParameter &)
99 return new SEService(user_data, listener);
102 SEService *SEService::createInstance(void *user_data,
103 serviceConnected handler)
104 throw(ErrorIO &, ErrorIllegalParameter &)
106 return new SEService(user_data, handler);
109 void SEService::reader_inserted(GObject *source_object,
110 guint reader_id, gchar *reader_name, gpointer user_data)
112 Reader *reader = NULL;
113 SEService *service = (SEService *)user_data;
115 _INFO("[MSG_NOTIFY_SE_INSERTED]");
118 reader = new Reader(service->context,
119 reader_name, GUINT_TO_POINTER(reader_id));
122 service->readers.push_back(reader);
126 _ERR("alloc failed");
129 if (service->listener != NULL)
131 service->listener->eventHandler(service,
132 reader_name, 1, service->context);
136 _DBG("listener is null");
140 void SEService::reader_removed(GObject *source_object,
141 guint reader_id, gchar *reader_name, gpointer user_data)
143 SEService *service = (SEService *)user_data;
146 _INFO("[MSG_NOTIFY_SE_REMOVED]");
148 for (i = 0; i < service->readers.size(); i++)
150 if (((Reader *)service->readers[i])->handle ==
151 GUINT_TO_POINTER(reader_id))
153 ((Reader *)service->readers[i])->unavailable();
158 if (service->listener != NULL)
160 service->listener->eventHandler(service,
161 reader_name, 2, service->context);
165 _DBG("listener is null");
169 void SEService::se_service_shutdown_cb(GObject *source_object,
170 GAsyncResult *res, gpointer user_data)
172 SEService *service = (SEService *)user_data;
174 GError *error = NULL;
176 if (smartcard_service_se_service_call_shutdown_finish(
177 SMARTCARD_SERVICE_SE_SERVICE(source_object),
178 &result, res, &error) == true) {
179 if (result == SCARD_ERROR_OK) {
180 service->connected = false;
182 _ERR("smartcard_service_se_service_call_shutdown failed, [%d]", result);
185 _ERR("smartcard_service_se_service_call_shutdown failed, [%s]", error->message);
190 void SEService::se_service_cb(GObject *source_object,
191 GAsyncResult *res, gpointer user_data)
193 SEService *service = (SEService *)user_data;
196 GVariant *readers = NULL;
197 GError *error = NULL;
199 if (service == NULL) {
200 _ERR("null parameter!!!");
204 if (smartcard_service_se_service_call_se_service_finish(
205 SMARTCARD_SERVICE_SE_SERVICE(source_object),
206 &result, &handle, &readers, res, &error) == true) {
207 if (result == SCARD_ERROR_OK) {
208 service->connected = true;
209 service->handle = handle;
210 service->parseReaderInformation(readers);
213 _ERR("smartcard_service_se_service_call_se_service failed, [%s]", error->message);
216 result = SCARD_ERROR_IPC_FAILED;
219 if (service->handler != NULL) {
220 service->handler(service, service->context);
221 } else if (service->listener != NULL) {
222 if (result == SCARD_ERROR_OK) {
223 service->listener->serviceConnected(service, service->context);
225 service->listener->errorHandler(service, result, service->context);
230 void SEService::shutdown()
235 void SEService::shutdownSync()
237 if (connected == true)
241 for (i = 0; i < readers.size(); i++)
243 readers[i]->closeSessions();
247 GError *error = NULL;
249 if (smartcard_service_se_service_call_shutdown_sync(
250 (SmartcardServiceSeService *)proxy,
255 _ERR("smartcard_service_se_service_call_shutdown_sync failed, [%s]", error->message);
260 /* wait at least 500ms */
261 usleep(SHUTDOWN_DELAY);
267 bool SEService::_initialize() throw(ErrorIO &)
273 /* init default context */
274 GError *error = NULL;
276 proxy = smartcard_service_se_service_proxy_new_for_bus_sync(
277 G_BUS_TYPE_SYSTEM, G_DBUS_PROXY_FLAGS_NONE,
278 "org.tizen.SmartcardService",
279 "/org/tizen/SmartcardService/SeService",
283 _ERR("Can not create proxy : %s", error->message);
288 g_signal_connect(proxy, "reader-inserted",
289 G_CALLBACK(&SEService::reader_inserted), this);
291 g_signal_connect(proxy, "reader-removed",
292 G_CALLBACK(&SEService::reader_removed), this);
295 smartcard_service_se_service_call_se_service(
296 (SmartcardServiceSeService *)proxy,
298 &SEService::se_service_cb,
306 int SEService::_initialize_sync_do_not_throw_exception()
310 GError *error = NULL;
311 GVariant *readers = NULL;
312 SEService *service = (SEService *)this;
316 /* init default context */
318 proxy = smartcard_service_se_service_proxy_new_for_bus_sync(
319 G_BUS_TYPE_SYSTEM, G_DBUS_PROXY_FLAGS_NONE,
320 "org.tizen.SmartcardService",
321 "/org/tizen/SmartcardService/SeService",
325 _ERR("Can not create proxy : %s", error->message);
330 g_signal_connect(proxy, "reader-inserted",
331 G_CALLBACK(&SEService::reader_inserted), this);
333 g_signal_connect(proxy, "reader-removed",
334 G_CALLBACK(&SEService::reader_removed), this);
337 if(smartcard_service_se_service_call_se_service_sync(
338 (SmartcardServiceSeService *)proxy, &result, &handle, &readers, NULL, &error) == true)
340 if (result == SCARD_ERROR_OK)
342 service->connected = true;
343 service->handle = handle;
344 service->parseReaderInformation(readers);
348 _ERR("Initialize error : %d", result);
352 if (service->handler != NULL) {
353 service->handler(service, service->context);
354 } else if (service->listener != NULL) {
355 if (result == SCARD_ERROR_OK) {
356 service->listener->serviceConnected(service, service->context);
358 service->listener->errorHandler(service, result, service->context);
367 int SEService::_initialize_sync() throw(ErrorIO &, ExceptionBase &)
371 GError *error = NULL;
372 GVariant *readers = NULL;
373 SEService *service = (SEService *)this;
377 /* init default context */
379 proxy = smartcard_service_se_service_proxy_new_for_bus_sync(
380 G_BUS_TYPE_SYSTEM, G_DBUS_PROXY_FLAGS_NONE,
381 "org.tizen.SmartcardService",
382 "/org/tizen/SmartcardService/SeService",
386 _ERR("Can not create proxy : %s", error->message);
391 g_signal_connect(proxy, "reader-inserted",
392 G_CALLBACK(&SEService::reader_inserted), this);
394 g_signal_connect(proxy, "reader-removed",
395 G_CALLBACK(&SEService::reader_removed), this);
398 if(smartcard_service_se_service_call_se_service_sync(
399 (SmartcardServiceSeService *)proxy, &result, &handle, &readers, NULL, &error) == true)
401 if (result == SCARD_ERROR_OK)
403 service->connected = true;
404 service->handle = handle;
405 service->parseReaderInformation(readers);
409 throw ExceptionBase(result);
414 _ERR("smartcard_service_se_service_call_se_service failed, [%s]", error->message);
417 result = SCARD_ERROR_IPC_FAILED;
425 bool SEService::initialize(void *context, serviceConnected handler)
426 throw(ErrorIO &, ErrorIllegalParameter &)
430 throw ErrorIllegalParameter(SCARD_ERROR_ILLEGAL_PARAM);
433 this->context = context;
434 this->handler = handler;
436 return _initialize();
439 bool SEService::initialize(void *context, SEServiceListener *listener)
440 throw(ErrorIO &, ErrorIllegalParameter &)
444 throw ErrorIllegalParameter(SCARD_ERROR_ILLEGAL_PARAM);
447 this->context = context;
448 this->listener = listener;
450 return _initialize_sync_do_not_throw_exception();
453 bool SEService::initializeSync(void *context)
454 throw(ErrorIO &, ErrorIllegalParameter &, ExceptionBase &)
456 this->context = context;
462 bool SEService::parseReaderInformation(GVariant *variant)
464 Reader *reader = NULL;
470 g_variant_get(variant, "a(us)", &iter);
472 while (g_variant_iter_loop(iter, "(us)", &handle, &name) == true)
474 SECURE_LOGD("Reader : name [%s], handle [%08x]", name, handle);
477 reader = new Reader((void *)this->handle, name, GUINT_TO_POINTER(handle));
480 _ERR("alloc failed");
484 readers.push_back(reader);
487 g_variant_iter_free(iter);
492 bool SEService::parseReaderInformation(unsigned int count,
493 const ByteArray &data)
496 unsigned int offset = 0;
497 unsigned int len = 0;
499 Reader *reader = NULL;
501 const uint8_t *buffer = NULL;
504 for (i = 0; i < count && offset < data.size(); i++)
506 memset(name, 0, sizeof(name));
508 buffer = data.getBuffer(offset);
512 memcpy(&len, buffer, sizeof(len));
513 offset += sizeof(len);
515 buffer = data.getBuffer(offset);
519 memcpy(name, buffer, len);
522 buffer = data.getBuffer(offset);
526 memcpy(&handle, buffer, sizeof(handle));
527 offset += sizeof(handle);
529 SECURE_LOGD("Reader [%d] : name [%s], handle [%p]", i, name, handle);
532 reader = new Reader(context, name, handle);
535 _ERR("alloc failed");
539 readers.push_back(reader);
544 } /* namespace smartcard_service_api */
547 #define SE_SERVICE_EXTERN_BEGIN \
548 if (handle != NULL) \
550 SEService *service = (SEService *)handle;
552 #define SE_SERVICE_EXTERN_END \
556 _ERR("Invalid param"); \
559 using namespace smartcard_service_api;
561 EXTERN_API se_service_h se_service_create_instance(void *user_data,
562 se_service_connected_cb callback)
568 service = new SEService(user_data, (serviceConnected)callback);
575 return (se_service_h)service;
578 EXTERN_API se_service_h se_service_create_instance_with_event_callback(
579 void *user_data, se_service_connected_cb connected,
580 se_service_event_cb event, se_sesrvice_error_cb error)
586 service = new SEService(user_data, (serviceConnected)connected);
593 return (se_service_h)service;
596 EXTERN_API se_service_h se_service_create_instance_sync(void *user_data,
603 service = new SEService(user_data);
605 catch (ExceptionBase &e)
607 *result = e.getErrorCode();
612 *result = SCARD_ERROR_UNKNOWN;
616 return (se_service_h)service;
619 EXTERN_API int se_service_get_version(se_service_h handle, char **version_str)
623 if (version_str == NULL) {
624 return SCARD_ERROR_ILLEGAL_PARAM;
627 SE_SERVICE_EXTERN_BEGIN;
629 *version_str = g_strdup(service->getVersion());
631 SE_SERVICE_EXTERN_END;
636 EXTERN_API int se_service_get_readers_count(se_service_h handle)
640 SE_SERVICE_EXTERN_BEGIN;
642 vector<ReaderHelper *> temp_readers;
644 temp_readers = service->getReaders();
645 count = temp_readers.size();
647 SE_SERVICE_EXTERN_END;
652 EXTERN_API int se_service_get_readers(se_service_h handle, int **readers, int *count)
656 SE_SERVICE_EXTERN_BEGIN;
658 vector<ReaderHelper *> temp_readers;
662 temp_readers = service->getReaders();
663 if(temp_readers.size() > 0)
665 *readers = (int *)calloc(temp_readers.size(), sizeof(int));
670 return SCARD_ERROR_NOT_ENOUGH_RESOURCE;
673 for (i = 0; i < temp_readers.size(); i++)
675 if (temp_readers[i]->isSecureElementPresent())
677 (*readers)[i] = (long)temp_readers[i];
684 SE_SERVICE_EXTERN_END;
689 EXTERN_API bool se_service_is_connected(se_service_h handle)
693 SE_SERVICE_EXTERN_BEGIN;
695 result = service->isConnected();
697 SE_SERVICE_EXTERN_END;
702 EXTERN_API void se_service_shutdown(se_service_h handle)
704 SE_SERVICE_EXTERN_BEGIN;
706 service->shutdownSync();
708 SE_SERVICE_EXTERN_END;
711 EXTERN_API int se_service_destroy_instance(se_service_h handle)
715 SE_SERVICE_EXTERN_BEGIN;
719 SE_SERVICE_EXTERN_END;