summaryrefslogtreecommitdiffstats
path: root/sys/dvb
diff options
context:
space:
mode:
Diffstat (limited to 'sys/dvb')
-rw-r--r--sys/dvb/Makefile.am39
-rw-r--r--sys/dvb/cam.c33
-rw-r--r--sys/dvb/cam.h33
-rw-r--r--sys/dvb/camapplication.c292
-rw-r--r--sys/dvb/camapplication.h84
-rw-r--r--sys/dvb/camapplicationinfo.c153
-rw-r--r--sys/dvb/camapplicationinfo.h41
-rw-r--r--sys/dvb/camconditionalaccess.c208
-rw-r--r--sys/dvb/camconditionalaccess.h55
-rw-r--r--sys/dvb/camdevice.c213
-rw-r--r--sys/dvb/camdevice.h69
-rw-r--r--sys/dvb/camresourcemanager.c192
-rw-r--r--sys/dvb/camresourcemanager.h41
-rw-r--r--sys/dvb/camsession.c595
-rw-r--r--sys/dvb/camsession.h96
-rw-r--r--sys/dvb/camswclient.c162
-rw-r--r--sys/dvb/camswclient.h53
-rw-r--r--sys/dvb/camtransport.c505
-rw-r--r--sys/dvb/camtransport.h115
-rw-r--r--sys/dvb/camutils.c340
-rw-r--r--sys/dvb/camutils.h59
-rw-r--r--sys/dvb/dvbbasebin.c897
-rw-r--r--sys/dvb/dvbbasebin.h75
-rw-r--r--sys/dvb/gstdvb.c46
-rw-r--r--sys/dvb/gstdvbsrc.c156
-rw-r--r--sys/dvb/gstdvbsrc.h5
26 files changed, 4483 insertions, 74 deletions
diff --git a/sys/dvb/Makefile.am b/sys/dvb/Makefile.am
index 33d090b6..1bcb099f 100644
--- a/sys/dvb/Makefile.am
+++ b/sys/dvb/Makefile.am
@@ -1,9 +1,36 @@
-plugin_LTLIBRARIES = libgstdvbsrc.la
+plugin_LTLIBRARIES = libgstdvb.la
-libgstdvbsrc_la_SOURCES = gstdvbsrc.c
-libgstdvbsrc_la_CFLAGS = $(GST_CFLAGS)
-libgstdvbsrc_la_LIBADD = $(GST_BASE_LIBS) $(GST_PLUGINS_BASE_LIBS)
-libgstdvbsrc_la_LDFLAGS = $(GST_PLUGIN_LDFLAGS)
+libgstdvb_la_SOURCES = \
+ gstdvb.c \
+ gstdvbsrc.c \
+ dvbbasebin.c \
+ cam.c \
+ camdevice.c \
+ camswclient.c \
+ camutils.c \
+ camtransport.c \
+ camsession.c \
+ camapplication.c \
+ camresourcemanager.c \
+ camapplicationinfo.c \
+ camconditionalaccess.c
-noinst_HEADERS = gstdvbsrc.h
+libgstdvb_la_CFLAGS = $(GST_CFLAGS)
+libgstdvb_la_LIBADD = $(GST_BASE_LIBS) $(GST_PLUGINS_BASE_LIBS)
+libgstdvb_la_LDFLAGS = $(GST_PLUGIN_LDFLAGS)
+
+noinst_HEADERS = \
+ gstdvbsrc.h \
+ dvbbasebin.h \
+ cam.h \
+ camdevice.h \
+ camswclient.h \
+ camutils.h \
+ camtransport.h \
+ camsession.h \
+ camapplication.h \
+ camresourcemanager.h \
+ camapplicationinfo.h \
+ camconditionalaccess.h
+
diff --git a/sys/dvb/cam.c b/sys/dvb/cam.c
new file mode 100644
index 00000000..ec67888d
--- /dev/null
+++ b/sys/dvb/cam.c
@@ -0,0 +1,33 @@
+/*
+ * cam.c -
+ * Copyright (C) 2007 Alessandro Decina
+ *
+ * Authors:
+ * Alessandro Decina <alessandro@nnva.org>
+ *
+ * This library is free software; you can redistribute it and/or
+ * modify it under the terms of the GNU Library General Public
+ * License as published by the Free Software Foundation; either
+ * version 2 of the License, or (at your option) any later version.
+ *
+ * This library 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
+ * Library General Public License for more details.
+ *
+ * You should have received a copy of the GNU Library General Public
+ * License along with this library; if not, write to the
+ * Free Software Foundation, Inc., 59 Temple Place - Suite 330,
+ * Boston, MA 02111-1307, USA.
+ */
+
+#include "cam.h"
+
+#define GST_CAT_DEFAULT cam_debug_cat
+GST_DEBUG_CATEGORY (cam_debug_cat);
+
+void
+cam_init ()
+{
+ GST_DEBUG_CATEGORY_INIT (cam_debug_cat, "dvbcam", 0, "DVB CAM support");
+}
diff --git a/sys/dvb/cam.h b/sys/dvb/cam.h
new file mode 100644
index 00000000..3211a2d1
--- /dev/null
+++ b/sys/dvb/cam.h
@@ -0,0 +1,33 @@
+/*
+ * cam.h -
+ * Copyright (C) 2007 Alessandro Decina
+ *
+ * Authors:
+ * Alessandro Decina <alessandro@nnva.org>
+ *
+ * This library is free software; you can redistribute it and/or
+ * modify it under the terms of the GNU Library General Public
+ * License as published by the Free Software Foundation; either
+ * version 2 of the License, or (at your option) any later version.
+ *
+ * This library 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
+ * Library General Public License for more details.
+ *
+ * You should have received a copy of the GNU Library General Public
+ * License along with this library; if not, write to the
+ * Free Software Foundation, Inc., 59 Temple Place - Suite 330,
+ * Boston, MA 02111-1307, USA.
+ */
+
+#ifndef CAM_H
+#define CAM_H
+
+#include <gst/gst.h>
+
+GST_DEBUG_CATEGORY_EXTERN (cam_debug_cat);
+
+void cam_init ();
+
+#endif /* CAM_H */
diff --git a/sys/dvb/camapplication.c b/sys/dvb/camapplication.c
new file mode 100644
index 00000000..03f9e552
--- /dev/null
+++ b/sys/dvb/camapplication.c
@@ -0,0 +1,292 @@
+/*
+ * camapplication.c - GStreamer CAM (EN50221) Application Layer
+ * Copyright (C) 2007 Alessandro Decina
+ *
+ * Authors:
+ * Alessandro Decina <alessandro@nnva.org>
+ *
+ * This library is free software; you can redistribute it and/or
+ * modify it under the terms of the GNU Library General Public
+ * License as published by the Free Software Foundation; either
+ * version 2 of the License, or (at your option) any later version.
+ *
+ * This library 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
+ * Library General Public License for more details.
+ *
+ * You should have received a copy of the GNU Library General Public
+ * License along with this library; if not, write to the
+ * Free Software Foundation, Inc., 59 Temple Place - Suite 330,
+ * Boston, MA 02111-1307, USA.
+ */
+
+#include "camapplication.h"
+
+#define GST_CAT_DEFAULT cam_debug_cat
+
+static CamReturn open_session_request_cb (CamSL * sl,
+ CamSLSession * session, CamSLResourceStatus * status);
+static CamReturn session_opened_cb (CamSL * sl, CamSLSession * session);
+static CamReturn session_closed_cb (CamSL * sl, CamSLSession * session);
+static CamReturn session_data_cb (CamSL * sl,
+ CamSLSession * session, guint8 * data, guint length);
+
+static guint
+resource_id_hash (gconstpointer key)
+{
+ guint resource_id = (guint) key;
+
+ if (!CAM_AL_RESOURCE_ID_IS_PUBLIC (resource_id)) {
+ /* private identifier, leave it as is */
+ return resource_id;
+ }
+
+ /* public identifier, mask out the version number */
+ return resource_id >> 6;
+}
+
+CamAL *
+cam_al_new (CamSL * sl)
+{
+ CamAL *al = g_new0 (CamAL, 1);
+
+ al->sl = sl;
+ al->applications = g_hash_table_new (resource_id_hash, g_direct_equal);
+
+ sl->user_data = al;
+ sl->open_session_request = open_session_request_cb;
+ sl->session_opened = session_opened_cb;
+ sl->session_closed = session_closed_cb;
+ sl->session_data = session_data_cb;
+
+ return al;
+}
+
+void
+cam_al_destroy (CamAL * al)
+{
+ g_hash_table_destroy (al->applications);
+ g_free (al);
+}
+
+gboolean
+cam_al_install (CamAL * al, CamALApplication * application)
+{
+ if (g_hash_table_lookup (al->applications,
+ GINT_TO_POINTER (application->resource_id)) != NULL)
+ return FALSE;
+
+ application->al = al;
+
+ g_hash_table_insert (al->applications,
+ GINT_TO_POINTER (application->resource_id), application);
+
+ return TRUE;
+}
+
+gboolean
+cam_al_uninstall (CamAL * al, CamALApplication * application)
+{
+ gboolean ret;
+
+ ret = g_hash_table_remove (al->applications,
+ GINT_TO_POINTER (application->resource_id));
+
+ return ret;
+}
+
+CamALApplication *
+cam_al_get (CamAL * al, guint resource_id)
+{
+ return CAM_AL_APPLICATION (g_hash_table_lookup (al->applications,
+ GINT_TO_POINTER (resource_id)));
+}
+
+void
+_cam_al_application_init (CamALApplication * application)
+{
+ application->sessions = NULL;
+}
+
+void
+_cam_al_application_destroy (CamALApplication * application)
+{
+ g_list_free (application->sessions);
+}
+
+static void
+foreach_get_key (gpointer key, gpointer value, gpointer user_data)
+{
+ GList **lst = (GList **) user_data;
+
+ *lst = g_list_append (*lst, key);
+}
+
+GList *
+cam_al_get_resource_ids (CamAL * al)
+{
+ GList *resource_ids = NULL;
+
+ g_hash_table_foreach (al->applications, foreach_get_key, &resource_ids);
+
+ return resource_ids;
+}
+
+void
+cam_al_calc_buffer_size (CamAL * al, guint body_length,
+ guint * buffer_size, guint * offset)
+{
+ guint apdu_header_length;
+ guint8 length_field_len;
+
+ /* get the length of the lenght_field() */
+ length_field_len = cam_calc_length_field_size (body_length);
+
+ /* sum the APDU header */
+ apdu_header_length = 3 + length_field_len;
+
+ /* chain up to the session layer to get the size of the buffer that can
+ * contain the whole APDU */
+ cam_sl_calc_buffer_size (al->sl, apdu_header_length + body_length,
+ buffer_size, offset);
+
+ /* add the APDU header to the SPDU offset */
+ *offset += apdu_header_length;
+}
+
+CamReturn
+cam_al_application_write (CamALApplication * application,
+ CamSLSession * session, guint tag, guint8 * buffer, guint buffer_size,
+ guint body_length)
+{
+ guint length_field_len;
+ guint apdu_header_length;
+ guint8 *apdu;
+
+ length_field_len = cam_calc_length_field_size (body_length);
+ apdu_header_length = 3 + length_field_len;
+ apdu = (buffer + buffer_size) - body_length - apdu_header_length;
+ apdu[0] = tag >> 16;
+ apdu[1] = (tag >> 8) & 0xFF;
+ apdu[2] = tag & 0xFF;
+
+ cam_write_length_field (&apdu[3], body_length);
+
+ return cam_sl_session_write (session, buffer, buffer_size,
+ apdu_header_length + body_length);
+}
+
+static CamReturn
+open_session_request_cb (CamSL * sl, CamSLSession * session,
+ CamSLResourceStatus * status)
+{
+ CamAL *al = CAM_AL (sl->user_data);
+ CamALApplication *application;
+ guint resource_id = session->resource_id;
+ CamReturn ret;
+
+ application = g_hash_table_lookup (al->applications,
+ GINT_TO_POINTER (resource_id));
+ if (application == NULL) {
+ *status = CAM_SL_RESOURCE_STATUS_NOT_FOUND;
+
+ return CAM_RETURN_OK;
+ }
+
+ if (CAM_AL_RESOURCE_ID_VERSION (application->resource_id)
+ < CAM_AL_RESOURCE_ID_VERSION (resource_id)) {
+ *status = CAM_SL_RESOURCE_STATUS_INVALID_VERSION;
+
+ return CAM_RETURN_OK;
+ }
+
+ ret = application->session_request (application, session, status);
+ if (CAM_FAILED (ret)) {
+ *status = CAM_SL_RESOURCE_STATUS_NOT_FOUND;
+
+ return ret;
+ }
+
+ if (*status == CAM_SL_RESOURCE_STATUS_OPEN) {
+ session->user_data = application;
+ application->sessions = g_list_append (application->sessions, session);
+ }
+
+ return CAM_RETURN_OK;
+}
+
+static CamReturn
+session_opened_cb (CamSL * sl, CamSLSession * session)
+{
+ CamALApplication *application;
+
+ application = CAM_AL_APPLICATION (session->user_data);
+ if (application == NULL) {
+ GST_ERROR ("session is established but has no application");
+ return CAM_RETURN_APPLICATION_ERROR;
+ }
+
+ return application->open (application, session);
+}
+
+static CamReturn
+session_closed_cb (CamSL * sl, CamSLSession * session)
+{
+ CamALApplication *application;
+ CamReturn ret;
+ GList *walk;
+
+ application = CAM_AL_APPLICATION (session->user_data);
+ if (application == NULL) {
+ GST_ERROR ("session is established but has no application");
+ return CAM_RETURN_APPLICATION_ERROR;
+ }
+
+ ret = application->close (application, session);
+ for (walk = application->sessions; walk; walk = g_list_next (walk)) {
+ CamSLSession *s = CAM_SL_SESSION (walk->data);
+
+ if (s->session_nb == session->session_nb) {
+ application->sessions = g_list_delete_link (application->sessions, walk);
+ break;
+ }
+ }
+
+ return ret;
+}
+
+static CamReturn
+session_data_cb (CamSL * sl, CamSLSession * session, guint8 * data, guint size)
+{
+ CamALApplication *application;
+ guint tag = 0;
+ guint8 length_field_len;
+ guint length;
+ guint i;
+
+ application = CAM_AL_APPLICATION (session->user_data);
+ if (application == NULL) {
+ GST_ERROR ("session is established but has no application");
+ return CAM_RETURN_APPLICATION_ERROR;
+ }
+
+ if (size < 4) {
+ GST_ERROR ("invalid APDU length %d", size);
+ return CAM_RETURN_APPLICATION_ERROR;
+ }
+
+ for (i = 0; i < 3; ++i)
+ tag = (tag << 8) | data[i];
+
+ length_field_len = cam_read_length_field (&data[3], &length);
+
+ if (length != size - 4) {
+ GST_ERROR ("unexpected APDU length %d expected %d", length, size);
+
+ return CAM_RETURN_APPLICATION_ERROR;
+ }
+
+ return application->data (application, session,
+ tag, data + 3 + length_field_len, length);
+}
diff --git a/sys/dvb/camapplication.h b/sys/dvb/camapplication.h
new file mode 100644
index 00000000..326d1f9b
--- /dev/null
+++ b/sys/dvb/camapplication.h
@@ -0,0 +1,84 @@
+/*
+ * camapplication.h - GStreamer CAM (EN50221) Application Layer
+ * Copyright (C) 2007 Alessandro Decina
+ *
+ * Authors:
+ * Alessandro Decina <alessandro@nnva.org>
+ *
+ * This library is free software; you can redistribute it and/or
+ * modify it under the terms of the GNU Library General Public
+ * License as published by the Free Software Foundation; either
+ * version 2 of the License, or (at your option) any later version.
+ *
+ * This library 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
+ * Library General Public License for more details.
+ *
+ * You should have received a copy of the GNU Library General Public
+ * License along with this library; if not, write to the
+ * Free Software Foundation, Inc., 59 Temple Place - Suite 330,
+ * Boston, MA 02111-1307, USA.
+ */
+
+#ifndef CAM_APPLICATION_LAYER_H
+#define CAM_APPLICATION_LAYER_H
+
+#include "cam.h"
+#include "camsession.h"
+
+#define CAM_AL(obj) ((CamAL *) obj)
+#define CAM_AL_APPLICATION(obj) ((CamALApplication *) obj)
+
+#define CAM_AL_RESOURCE_ID_IS_PUBLIC(resource_id) ((resource_id >> 30) != 3)
+#define CAM_AL_RESOURCE_ID_CLASS(resource_id) ((resource_id >> 16) & 0x3FFF)
+#define CAM_AL_RESOURCE_ID_TYPE(resource_id) ((resource_id >> 6) & 0x03FF)
+#define CAM_AL_RESOURCE_ID_VERSION(resource_id) (resource_id & 0x3F)
+
+#define CAM_AL_RESOURCE_MANAGER_ID 0x10041
+#define CAM_AL_APPLICATION_INFO_ID 0x20041
+#define CAM_AL_CONDITIONAL_ACCESS_ID 0x30041
+
+typedef struct _CamAL CamAL;
+typedef struct _CamALApplication CamALApplication;
+
+struct _CamAL
+{
+ CamSL *sl;
+
+ GHashTable *applications;
+};
+
+struct _CamALApplication
+{
+ CamAL *al;
+ guint resource_id;
+ GList *sessions;
+
+ /* vtable */
+ CamReturn (*session_request) (CamALApplication *application,
+ CamSLSession *session, CamSLResourceStatus *status);
+ CamReturn (*open) (CamALApplication *application, CamSLSession *session);
+ CamReturn (*close) (CamALApplication *application, CamSLSession *session);
+ CamReturn (*data) (CamALApplication *application, CamSLSession *session,
+ guint tag, guint8 *buffer, guint length);
+};
+
+CamAL *cam_al_new (CamSL *sl);
+void cam_al_destroy (CamAL *al);
+
+gboolean cam_al_install (CamAL *al, CamALApplication *application);
+gboolean cam_al_uninstall (CamAL *al, CamALApplication *application);
+CamALApplication *cam_al_get (CamAL *al, guint resource_id);
+GList *cam_al_get_resource_ids (CamAL *al);
+
+void cam_al_calc_buffer_size (CamAL *al, guint body_length,
+ guint *buffer_size, guint *offset);
+
+void _cam_al_application_init (CamALApplication *application);
+void _cam_al_application_destroy (CamALApplication *application);
+
+CamReturn cam_al_application_write (CamALApplication *application,
+ CamSLSession *session, guint tag, guint8 *buffer,
+ guint buffer_size, guint body_length);
+#endif /* CAM_APPLICATION_LAYER_H */
diff --git a/sys/dvb/camapplicationinfo.c b/sys/dvb/camapplicationinfo.c
new file mode 100644
index 00000000..63a885a1
--- /dev/null
+++ b/sys/dvb/camapplicationinfo.c
@@ -0,0 +1,153 @@
+/*
+ * camapplicationinfo.h - CAM (EN50221) Application Info resource
+ * Copyright (C) 2007 Alessandro Decina
+ *
+ * Authors:
+ * Alessandro Decina <alessandro@nnva.org>
+ *
+ * This library is free software; you can redistribute it and/or
+ * modify it under the terms of the GNU Library General Public
+ * License as published by the Free Software Foundation; either
+ * version 2 of the License, or (at your option) any later version.
+ *
+ * This library 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
+ * Library General Public License for more details.
+ *
+ * You should have received a copy of the GNU Library General Public
+ * License along with this library; if not, write to the
+ * Free Software Foundation, Inc., 59 Temple Place - Suite 330,
+ * Boston, MA 02111-1307, USA.
+ */
+
+#include <string.h>
+
+#include "camapplicationinfo.h"
+
+#define GST_CAT_DEFAULT cam_debug_cat
+#define TAG_APPLICATION_INFO_ENQUIRY 0x9F8020
+#define TAG_APPLICATION_INFO_REPLY 0x9F8021
+#define TAG_APPLICATION_INFO_ENTER_MENU 0x9F8022
+
+static CamReturn session_request_impl (CamALApplication * application,
+ CamSLSession * session, CamSLResourceStatus * status);
+static CamReturn open_impl (CamALApplication * application,
+ CamSLSession * session);
+static CamReturn close_impl (CamALApplication * application,
+ CamSLSession * session);
+static CamReturn data_impl (CamALApplication * application,
+ CamSLSession * session, guint tag, guint8 * buffer, guint length);
+
+CamApplicationInfo *
+cam_application_info_new ()
+{
+ CamApplicationInfo *info;
+ CamALApplication *application;
+
+ info = g_new0 (CamApplicationInfo, 1);
+ application = CAM_AL_APPLICATION (info);
+ _cam_al_application_init (application);
+ application->resource_id = CAM_AL_APPLICATION_INFO_ID;
+ application->session_request = session_request_impl;
+ application->open = open_impl;
+ application->close = close_impl;
+ application->data = data_impl;
+
+ return info;
+}
+
+void
+cam_application_info_destroy (CamApplicationInfo * info)
+{
+ _cam_al_application_destroy (CAM_AL_APPLICATION (info));
+ g_free (info);
+}
+
+static CamReturn
+send_simple (CamApplicationInfo * info, CamSLSession * session, guint tag)
+{
+ guint8 *buffer;
+ guint offset;
+ guint buffer_size;
+ CamReturn ret;
+
+ cam_al_calc_buffer_size (CAM_AL_APPLICATION (info)->al, 0, &buffer_size,
+ &offset);
+ buffer = g_malloc (buffer_size);
+
+ ret = cam_al_application_write (CAM_AL_APPLICATION (info), session,
+ tag, buffer, buffer_size, 0);
+
+ g_free (buffer);
+
+ return ret;
+}
+
+static CamReturn
+send_application_info_enquiry (CamApplicationInfo * info,
+ CamSLSession * session)
+{
+ GST_DEBUG ("sending application info enquiry");
+ return send_simple (info, session, TAG_APPLICATION_INFO_ENQUIRY);
+}
+
+static CamReturn
+session_request_impl (CamALApplication * application,
+ CamSLSession * session, CamSLResourceStatus * status)
+{
+ *status = CAM_SL_RESOURCE_STATUS_OPEN;
+
+ return CAM_RETURN_OK;
+}
+
+static CamReturn
+open_impl (CamALApplication * application, CamSLSession * session)
+{
+ CamApplicationInfo *info = CAM_APPLICATION_INFO (application);
+
+ return send_application_info_enquiry (info, session);
+}
+
+static CamReturn
+close_impl (CamALApplication * application, CamSLSession * session)
+{
+ return CAM_RETURN_OK;
+}
+
+static CamReturn
+handle_application_info_reply (CamApplicationInfo * info,
+ CamSLSession * session, guint8 * buffer, guint length)
+{
+ guint8 type;
+ guint8 menu_length;
+ gchar menu[255];
+
+ type = buffer[0];
+ menu_length = buffer[5];
+ menu_length = MIN (menu_length, 255);
+ memcpy (menu, buffer + 6, menu_length);
+ menu[menu_length] = 0;
+
+ GST_INFO ("application info reply, type: %d, menu: %s", type, menu);
+
+ return CAM_RETURN_OK;
+}
+
+static CamReturn
+data_impl (CamALApplication * application, CamSLSession * session,
+ guint tag, guint8 * buffer, guint length)
+{
+ CamReturn ret;
+ CamApplicationInfo *info = CAM_APPLICATION_INFO (application);
+
+ switch (tag) {
+ case TAG_APPLICATION_INFO_REPLY:
+ ret = handle_application_info_reply (info, session, buffer, length);
+ break;
+ default:
+ g_return_val_if_reached (CAM_RETURN_ERROR);
+ }
+
+ return ret;
+}
diff --git a/sys/dvb/camapplicationinfo.h b/sys/dvb/camapplicationinfo.h
new file mode 100644
index 00000000..c8aa8fab
--- /dev/null
+++ b/sys/dvb/camapplicationinfo.h
@@ -0,0 +1,41 @@
+/*
+ * camapplicationinfo.h - CAM (EN50221) Application Info resource
+ * Copyright (C) 2007 Alessandro Decina
+ *
+ * Authors:
+ * Alessandro Decina <alessandro@nnva.org>
+ *
+ * This library is free software; you can redistribute it and/or
+ * modify it under the terms of the GNU Library General Public
+ * License as published by the Free Software Foundation; either
+ * version 2 of the License, or (at your option) any later version.
+ *
+ * This library 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
+ * Library General Public License for more details.
+ *
+ * You should have received a copy of the GNU Library General Public
+ * License along with this library; if not, write to the
+ * Free Software Foundation, Inc., 59 Temple Place - Suite 330,
+ * Boston, MA 02111-1307, USA.
+ */
+
+#ifndef CAM_APPLICATION_INFO_H
+#define CAM_APPLICATION_INFO_H
+
+#include "camapplication.h"
+
+#define CAM_APPLICATION_INFO(obj) ((CamApplicationInfo *) obj)
+
+typedef struct _CamApplicationInfo CamApplicationInfo;
+
+struct _CamApplicationInfo
+{
+ CamALApplication application;
+};
+
+CamApplicationInfo *cam_application_info_new ();
+void cam_application_info_destroy (CamApplicationInfo *info);
+
+#endif /* CAM_APPLICATION_INFO_H */
diff --git a/sys/dvb/camconditionalaccess.c b/sys/dvb/camconditionalaccess.c
new file mode 100644
index 00000000..985ce293
--- /dev/null
+++ b/sys/dvb/camconditionalaccess.c
@@ -0,0 +1,208 @@
+/*
+ * camconditionalaccess.c - CAM (EN50221) Conditional Access resource
+ * Copyright (C) 2007 Alessandro Decina
+ *
+ * Authors:
+ * Alessandro Decina <alessandro@nnva.org>
+ *
+ * This program 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 2 of the License, or
+ * (at your option) any later version.
+ *
+ * This program 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 this program; if not, write to the Free Software
+ * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
+ *
+ */
+
+#include <unistd.h>
+#include <string.h>
+#include "camutils.h"
+#include "camconditionalaccess.h"
+
+#define GST_CAT_DEFAULT cam_debug_cat
+#define TAG_CONDITIONAL_ACCESS_INFO_ENQUIRY 0x9F8030
+#define TAG_CONDITIONAL_ACCESS_INFO_REPLY 0x9F8031
+#define TAG_CONDITIONAL_ACCESS_PMT 0x9F8032
+
+static CamReturn session_request_impl (CamALApplication * application,
+ CamSLSession * session, CamSLResourceStatus * status);
+static CamReturn open_impl (CamALApplication * application,
+ CamSLSession * session);
+static CamReturn close_impl (CamALApplication * application,
+ CamSLSession * session);
+static CamReturn data_impl (CamALApplication * application,
+ CamSLSession * session, guint tag, guint8 * buffer, guint length);
+
+CamConditionalAccess *
+cam_conditional_access_new ()
+{
+ CamConditionalAccess *cas;
+ CamALApplication *application;
+
+ cas = g_new0 (CamConditionalAccess, 1);
+
+ application = CAM_AL_APPLICATION (cas);
+ _cam_al_application_init (application);
+ application->resource_id = CAM_AL_CONDITIONAL_ACCESS_ID;
+ application->session_request = session_request_impl;
+ application->open = open_impl;
+ application->close = close_impl;
+ application->data = data_impl;
+
+ cas->ready = FALSE;
+
+ return cas;
+}
+
+void
+cam_conditional_access_destroy (CamConditionalAccess * cas)
+{
+ _cam_al_application_destroy (CAM_AL_APPLICATION (cas));
+ g_free (cas);
+}
+
+static CamReturn
+send_ca_pmt (CamConditionalAccess * cas, GObject * pmt,
+ guint8 list_management, guint8 cmd_id)
+{
+ CamReturn ret;
+ guint8 *buffer;
+ guint buffer_size;
+ guint offset;
+ guint8 *ca_pmt;
+ guint ca_pmt_size;
+ GList *walk;
+
+ ca_pmt = cam_build_ca_pmt (pmt, list_management, cmd_id, &ca_pmt_size);
+ cam_al_calc_buffer_size (CAM_AL_APPLICATION (cas)->al,
+ ca_pmt_size, &buffer_size, &offset);
+
+ buffer = g_malloc0 (buffer_size);
+ memcpy (buffer + offset, ca_pmt, ca_pmt_size);
+
+ for (walk = CAM_AL_APPLICATION (cas)->sessions; walk; walk = walk->next) {
+ CamSLSession *session = CAM_SL_SESSION (walk->data);
+
+ ret = cam_al_application_write (CAM_AL_APPLICATION (cas), session,
+ TAG_CONDITIONAL_ACCESS_PMT, buffer, buffer_size, ca_pmt_size);
+ if (CAM_FAILED (ret)) {
+ GST_ERROR ("error sending ca_pmt to slot %d, error: %d",
+ session->connection->slot, ret);
+ continue;
+ }
+ }
+
+ g_free (ca_pmt);
+ g_free (buffer);
+
+ return CAM_RETURN_OK;
+}
+
+CamReturn
+cam_conditional_access_set_pmt (CamConditionalAccess * cas,
+ GObject * pmt, CamConditionalAccessPmtFlag flag)
+{
+ return send_ca_pmt (cas, pmt, flag, 0x01 /* ok_descrambling */ );
+}
+
+static CamReturn
+send_simple (CamConditionalAccess * cas, CamSLSession * session, guint tag)
+{
+ guint8 *buffer;
+ guint offset;
+ guint buffer_size;
+ CamReturn ret;
+
+ cam_al_calc_buffer_size (CAM_AL_APPLICATION (cas)->al, 0, &buffer_size,
+ &offset);
+ buffer = g_malloc (buffer_size);
+
+ ret = cam_al_application_write (CAM_AL_APPLICATION (cas), session,
+ tag, buffer, buffer_size, 0);
+
+ g_free (buffer);
+
+ return ret;
+}
+
+static CamReturn
+send_conditional_access_enquiry (CamConditionalAccess * cas,
+ CamSLSession * session)
+{
+ GST_DEBUG ("sending application cas enquiry");
+ return send_simple (cas, session, TAG_CONDITIONAL_ACCESS_INFO_ENQUIRY);
+}
+
+static CamReturn
+session_request_impl (CamALApplication * application,
+ CamSLSession * session, CamSLResourceStatus * status)
+{
+ *status = CAM_SL_RESOURCE_STATUS_OPEN;
+
+ return CAM_RETURN_OK;
+}
+
+static CamReturn
+open_impl (CamALApplication * application, CamSLSession * session)
+{
+ CamConditionalAccess *cas = CAM_CONDITIONAL_ACCESS (application);
+
+ GST_INFO ("opening conditional access session %d", session->session_nb);
+
+ return send_conditional_access_enquiry (cas, session);
+}
+
+static CamReturn
+close_impl (CamALApplication * application, CamSLSession * session)
+{
+ GST_INFO ("closing conditional access session %d", session->session_nb);
+
+ return CAM_RETURN_OK;
+}
+
+static CamReturn
+handle_conditional_access_info_reply (CamConditionalAccess * cas,
+ CamSLSession * session, guint8 * buffer, guint length)
+{
+ int i;
+ guint16 cas_id;
+
+ GST_INFO ("conditional access info enquiry reply");
+
+ for (i = 0; i < length / 2; ++i) {
+ cas_id = GST_READ_UINT16_BE (buffer);
+
+ GST_INFO ("slot %d, cas_id 0x%x", session->connection->slot, cas_id);
+
+ buffer += 2;
+ }
+
+ cas->ready = TRUE;
+
+ return CAM_RETURN_OK;
+}
+
+static CamReturn
+data_impl (CamALApplication * application, CamSLSession * session,
+ guint tag, guint8 * buffer, guint length)
+{
+ CamReturn ret;
+ CamConditionalAccess *cas = CAM_CONDITIONAL_ACCESS (application);
+
+ switch (tag) {
+ case TAG_CONDITIONAL_ACCESS_INFO_REPLY:
+ ret = handle_conditional_access_info_reply (cas, session, buffer, length);
+ break;
+ default:
+ g_return_val_if_reached (CAM_RETURN_ERROR);
+ }
+
+ return ret;
+}
diff --git a/sys/dvb/camconditionalaccess.h b/sys/dvb/camconditionalaccess.h
new file mode 100644
index 00000000..82c54d17
--- /dev/null
+++ b/sys/dvb/camconditionalaccess.h
@@ -0,0 +1,55 @@
+/*
+ * camconditionalaccess.h - CAM (EN50221) Conditional Access resource
+ * Copyright (C) 2007 Alessandro Decina
+ *
+ * Authors:
+ * Alessandro Decina <alessandro@nnva.org>
+ *
+ * This program 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 2 of the License, or
+ * (at your option) any later version.
+ *
+ * This program 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 this program; if not, write to the Free Software
+ * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
+ *
+ */
+
+#ifndef CAM_CONDITIONAL_ACCESS_H
+#define CAM_CONDITIONAL_ACCESS_H
+
+#include "camapplication.h"
+
+#define CAM_CONDITIONAL_ACCESS(obj) ((CamConditionalAccess *) obj)
+
+typedef enum
+{
+ CAM_CONDITIONAL_ACCESS_PMT_FLAG_MORE = 0,
+ CAM_CONDITIONAL_ACCESS_PMT_FLAG_FIRST,
+ CAM_CONDITIONAL_ACCESS_PMT_FLAG_LAST,
+ CAM_CONDITIONAL_ACCESS_PMT_FLAG_ONLY,
+ CAM_CONDITIONAL_ACCESS_PMT_FLAG_ADD,
+ CAM_CONDITIONAL_ACCESS_PMT_FLAG_UPDATE,
+} CamConditionalAccessPmtFlag;
+
+typedef struct _CamConditionalAccess CamConditionalAccess;
+
+struct _CamConditionalAccess
+{
+ CamALApplication application;
+ gboolean ready;
+};
+
+CamConditionalAccess *cam_conditional_access_new ();
+void cam_conditional_access_destroy (CamConditionalAccess *cas);
+
+CamReturn cam_conditional_access_set_pmt (CamConditionalAccess *cas,
+ GObject *pmt, CamConditionalAccessPmtFlag flag);
+
+#endif /* CAM_CONDITIONAL_ACCESS_H */
diff --git a/sys/dvb/camdevice.c b/sys/dvb/camdevice.c
new file mode 100644
index 00000000..174edca4
--- /dev/null
+++ b/sys/dvb/camdevice.c
@@ -0,0 +1,213 @@
+/*
+ * camdevice.c - GStreamer hardware CAM support
+ * Copyright (C) 2007 Alessandro Decina
+ *
+ * Authors:
+ * Alessandro Decina <alessandro@nnva.org>
+ *
+ * This library is free software; you can redistribute it and/or
+ * modify it under the terms of the GNU Library General Public
+ * License as published by the Free Software Foundation; either
+ * version 2 of the License, or (at your option) any later version.
+ *
+ * This library 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
+ * Library General Public License for more details.
+ *
+ * You should have received a copy of the GNU Library General Public
+ * License along with this library; if not, write to the
+ * Free Software Foundation, Inc., 59 Temple Place - Suite 330,
+ * Boston, MA 02111-1307, USA.
+ */
+
+#include <glib.h>
+#include <sys/ioctl.h>
+#include <fcntl.h>
+#include <unistd.h>
+#include <string.h>
+#include <linux/dvb/ca.h>
+
+#include "camdevice.h"
+
+#define GST_CAT_DEFAULT cam_debug_cat
+
+CamDevice *
+cam_device_new ()
+{
+ CamDevice *device = g_new0 (CamDevice, 1);
+
+ device->state = CAM_DEVICE_STATE_CLOSED;
+
+ return device;
+}
+
+static void
+reset_state (CamDevice * device)
+{
+ if (device->filename) {
+ g_free (device->filename);
+ device->filename = NULL;
+ }
+
+ if (device->fd) {
+ close (device->fd);
+ device->fd = -1;
+ }
+
+ if (device->cas) {
+ cam_conditional_access_destroy (device->cas);
+ device->cas = NULL;
+ }
+
+ if (device->mgr) {
+ cam_resource_manager_destroy (device->mgr);
+ device->mgr = NULL;
+ }
+
+ if (device->info) {
+ cam_application_info_destroy (device->info);
+ device->info = NULL;
+ }
+
+ if (device->al) {
+ cam_al_destroy (device->al);
+ device->al = NULL;
+ }
+
+ if (device->sl) {
+ cam_sl_destroy (device->sl);
+ device->sl = NULL;
+ }
+
+ if (device->tl) {
+ cam_tl_destroy (device->tl);
+ device->tl = NULL;
+ }
+
+ device->state = CAM_DEVICE_STATE_CLOSED;
+}
+
+void
+cam_device_free (CamDevice * device)
+{
+ if (device->state != CAM_DEVICE_STATE_CLOSED)
+ GST_WARNING ("device not in CLOSED state when free'd");
+
+ reset_state (device);
+ g_free (device);
+}
+
+gboolean
+cam_device_open (CamDevice * device, const char *filename)
+{
+ ca_caps_t ca_caps;
+ int ret;
+ int i;
+
+ g_return_val_if_fail (device != NULL, FALSE);
+ g_return_val_if_fail (device->state == CAM_DEVICE_STATE_CLOSED, FALSE);
+ g_return_val_if_fail (filename != NULL, FALSE);
+
+ GST_INFO ("opening ca device %s", filename);
+
+ ret = open (filename, O_RDWR);
+ if (ret == -1) {
+ GST_ERROR ("can't open ca device: %s", strerror (errno));
+ return FALSE;
+ }
+
+ device->fd = ret;
+
+ ret = ioctl (device->fd, CA_RESET);
+ sleep (1);
+
+ /* get the capabilities of the CA */
+ ret = ioctl (device->fd, CA_GET_CAP, &ca_caps);
+ if (ret == -1) {
+ GST_ERROR ("CA_GET_CAP ioctl failed: %s", strerror (errno));
+ reset_state (device);
+ return FALSE;
+ }
+
+ device->tl = cam_tl_new (device->fd);
+ device->sl = cam_sl_new (device->tl);
+ device->al = cam_al_new (device->sl);
+
+ device->mgr = cam_resource_manager_new ();
+ cam_al_install (device->al, CAM_AL_APPLICATION (device->mgr));
+
+ device->info = cam_application_info_new ();
+ cam_al_install (device->al, CAM_AL_APPLICATION (device->info));
+
+ device->cas = cam_conditional_access_new ();
+ cam_al_install (device->al, CAM_AL_APPLICATION (device->cas));
+
+ /* open a connection to each slot */
+ for (i = 0; i < ca_caps.slot_num; ++i) {
+ CamTLConnection *connection;
+
+ ret = cam_tl_create_connection (device->tl, i, &connection);
+ if (CAM_FAILED (ret)) {
+ /* just ignore the slot, error out later only if no connection has been
+ * established */
+ GST_WARNING ("connection to slot %d failed, error: %d", i, ret);
+ continue;
+ }
+ }
+
+ if (g_hash_table_size (device->tl->connections) == 0) {
+ GST_ERROR ("couldn't connect to any slot");
+
+ reset_state (device);
+ return FALSE;
+ }
+
+ device->state = CAM_DEVICE_STATE_OPEN;
+ device->filename = g_strdup (filename);
+
+ /* poll each connection to initiate the protocol */
+ cam_tl_read_all (device->tl, TRUE);
+
+ return TRUE;
+}
+
+void
+cam_device_close (CamDevice * device)
+{
+ g_return_if_fail (device != NULL);
+ g_return_if_fail (device->state == CAM_DEVICE_STATE_OPEN);
+
+ GST_INFO ("closing ca device %s", device->filename);
+ reset_state (device);
+}
+
+void
+cam_device_poll (CamDevice * device)
+{
+ g_return_if_fail (device != NULL);
+ g_return_if_fail (device->state == CAM_DEVICE_STATE_OPEN);
+
+ cam_tl_read_all (device->tl, TRUE);
+}
+
+gboolean
+cam_device_ready (CamDevice * device)
+{
+ g_return_val_if_fail (device != NULL, FALSE);
+ g_return_val_if_fail (device->state == CAM_DEVICE_STATE_OPEN, FALSE);
+
+ return device->cas->ready;
+}
+
+void
+cam_device_set_pmt (CamDevice * device,
+ GObject * pmt, CamConditionalAccessPmtFlag flag)
+{
+ g_return_if_fail (device != NULL);
+ g_return_if_fail (device->state == CAM_DEVICE_STATE_OPEN);
+ g_return_if_fail (pmt != NULL);
+
+ cam_conditional_access_set_pmt (device->cas, pmt, flag);
+ cam_tl_read_all (device->tl, FALSE);
+}
diff --git a/sys/dvb/camdevice.h b/sys/dvb/camdevice.h
new file mode 100644
index 00000000..2dbc79df
--- /dev/null
+++ b/sys/dvb/camdevice.h
@@ -0,0 +1,69 @@
+/*
+ * camdevice.h - GStreamer hardware CAM interface
+ * Copyright (C) 2007 Alessandro Decina
+ *
+ * Authors:
+ * Alessandro Decina <alessandro@nnva.org>
+ *
+ * This library is free software; you can redistribute it and/or
+ * modify it under the terms of the GNU Library General Public
+ * License as published by the Free Software Foundation; either
+ * version 2 of the License, or (at your option) any later version.
+ *
+ * This library 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
+ * Library General Public License for more details.
+ *
+ * You should have received a copy of the GNU Library General Public
+ * License along with this library; if not, write to the
+ * Free Software Foundation, Inc., 59 Temple Place - Suite 330,
+ * Boston, MA 02111-1307, USA.
+ */
+
+#ifndef CAM_DEVICE_H
+#define CAM_DEVICE_H
+
+#include "camtransport.h"
+#include "camsession.h"
+#include "camapplication.h"
+#include "camresourcemanager.h"
+#include "camapplicationinfo.h"
+#include "camconditionalaccess.h"
+
+typedef enum
+{
+ CAM_DEVICE_STATE_CLOSED,
+ CAM_DEVICE_STATE_OPEN,
+} CamDeviceState;
+
+typedef struct
+{
+ /* private */
+ CamDeviceState state;
+ char *filename;
+ int fd;
+
+ /* EN50221 layers */
+ CamTL *tl;
+ CamSL *sl;
+ CamAL *al;
+
+ /* apps provided by us */
+ CamResourceManager *mgr;
+ CamApplicationInfo *info;
+ CamConditionalAccess *cas;
+} CamDevice;
+
+CamDevice *cam_device_new ();
+void cam_device_free (CamDevice *device);
+
+gboolean cam_device_open (CamDevice *device, const char *filename);
+void cam_device_close (CamDevice *device);
+
+gboolean cam_device_ready (CamDevice *device);
+void cam_device_poll (CamDevice *device);
+void cam_device_set_pmt (CamDevice *device,
+ GObject *pmt, CamConditionalAccessPmtFlag flag);
+
+#endif /* CAM_DEVICE_H */
diff --git a/sys/dvb/camresourcemanager.c b/sys/dvb/camresourcemanager.c
new file mode 100644
index 00000000..8b7ef3b4
--- /dev/null
+++ b/sys/dvb/camresourcemanager.c
@@ -0,0 +1,192 @@
+/*
+ * camresourcemanager.c - GStreamer CAM (EN50221) Resource Manager
+ * Copyright (C) 2007 Alessandro Decina
+ *
+ * Authors:
+ * Alessandro Decina <alessandro@nnva.org>
+ *
+ * This library is free software; you can redistribute it and/or
+ * modify it under the terms of the GNU Library General Public
+ * License as published by the Free Software Foundation; either
+ * version 2 of the License, or (at your option) any later version.
+ *
+ * This library 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
+ * Library General Public License for more details.
+ *
+ * You should have received a copy of the GNU Library General Public
+ * License along with this library; if not, write to the
+ * Free Software Foundation, Inc., 59 Temple Place - Suite 330,
+ * Boston, MA 02111-1307, USA.
+ */
+
+#include "camresourcemanager.h"
+
+#define GST_CAT_DEFAULT cam_debug_cat
+#define TAG_PROFILE_ENQUIRY 0x9F8010
+#define TAG_PROFILE_REPLY 0x9F8011
+#define TAG_PROFILE_CHANGE 0x9F8012
+
+static CamReturn session_request_impl (CamALApplication * application,
+ CamSLSession * session, CamSLResourceStatus * status);
+static CamReturn open_impl (CamALApplication * application,
+ CamSLSession * session);
+static CamReturn close_impl (CamALApplication * application,
+ CamSLSession * session);
+static CamReturn data_impl (CamALApplication * application,
+ CamSLSession * session, guint tag, guint8 * buffer, guint length);
+
+CamResourceManager *
+cam_resource_manager_new ()
+{
+ CamALApplication *application;
+ CamResourceManager *mgr;
+
+ mgr = g_new0 (CamResourceManager, 1);
+ application = CAM_AL_APPLICATION (mgr);
+ _cam_al_application_init (application);
+ application->resource_id = CAM_AL_RESOURCE_MANAGER_ID;
+ application->session_request = session_request_impl;
+ application->open = open_impl;
+ application->close = close_impl;
+ application->data = data_impl;
+
+ return mgr;
+}
+
+void
+cam_resource_manager_destroy (CamResourceManager * mgr)
+{
+ _cam_al_application_destroy (CAM_AL_APPLICATION (mgr));
+ g_free (mgr);
+}
+
+static CamReturn
+session_request_impl (CamALApplication * application,
+ CamSLSession * session, CamSLResourceStatus * status)
+{
+ *status = CAM_SL_RESOURCE_STATUS_OPEN;
+
+ return CAM_RETURN_OK;
+}
+
+static CamReturn
+send_simple (CamResourceManager * mgr, CamSLSession * session, guint tag)
+{
+ guint8 *buffer;
+ guint offset;
+ guint buffer_size;
+ CamReturn ret;
+
+ cam_al_calc_buffer_size (CAM_AL_APPLICATION (mgr)->al, 0, &buffer_size,
+ &offset);
+ buffer = g_malloc (buffer_size);
+
+ ret = cam_al_application_write (CAM_AL_APPLICATION (mgr), session,
+ tag, buffer, buffer_size, 0);
+
+ g_free (buffer);
+
+ return ret;
+}
+
+static CamReturn
+send_profile_enquiry (CamResourceManager * mgr, CamSLSession * session)
+{
+ GST_DEBUG ("sending profile enquiry");
+ return send_simple (mgr, session, TAG_PROFILE_ENQUIRY);
+}
+
+static CamReturn
+send_profile_change (CamResourceManager * mgr, CamSLSession * session)
+{
+ GST_DEBUG ("sending profile change");
+ return send_simple (mgr, session, TAG_PROFILE_CHANGE);
+}
+
+static CamReturn
+send_profile_reply (CamResourceManager * mgr, CamSLSession * session)
+{
+ CamReturn ret;
+ guint8 *buffer;
+ guint8 *apdu_body;
+ guint buffer_size;
+ guint offset;
+ GList *resource_ids;
+ guint resource_ids_size;
+ GList *walk;
+
+ resource_ids = cam_al_get_resource_ids (CAM_AL_APPLICATION (mgr)->al);
+ resource_ids_size = g_list_length (resource_ids) * 4;
+
+ cam_al_calc_buffer_size (CAM_AL_APPLICATION (mgr)->al, resource_ids_size,
+ &buffer_size, &offset);
+
+ buffer = g_malloc (buffer_size);
+ apdu_body = buffer + offset;
+
+ for (walk = resource_ids; walk != NULL; walk = walk->next) {
+ GST_WRITE_UINT32_BE (apdu_body, walk->data);
+
+ apdu_body += 4;
+ }
+
+ g_list_free (resource_ids);
+
+ GST_DEBUG ("sending profile reply");
+ ret = cam_al_application_write (CAM_AL_APPLICATION (mgr), session,
+ TAG_PROFILE_REPLY, buffer, buffer_size, resource_ids_size);
+
+ g_free (buffer);
+
+ return ret;
+}
+
+static CamReturn
+open_impl (CamALApplication * application, CamSLSession * session)
+{
+ CamResourceManager *mgr = CAM_RESOURCE_MANAGER (application);
+
+ return send_profile_enquiry (mgr, session);
+}
+
+static CamReturn
+close_impl (CamALApplication * application, CamSLSession * session)
+{
+ return CAM_RETURN_OK;
+}
+
+static CamReturn
+handle_profile_reply (CamResourceManager * mgr, CamSLSession * session,
+ guint8 * buffer, guint length)
+{
+ /* the APDU contains length / 4 resource_ids */
+ /* FIXME: implement this once CamApplicationProxy is implemented */
+ GST_DEBUG ("got profile reply");
+ return send_profile_change (mgr, session);
+}
+
+static CamReturn
+data_impl (CamALApplication * application, CamSLSession * session,
+ guint tag, guint8 * buffer, guint length)
+{
+ CamReturn ret;
+ CamResourceManager *mgr = CAM_RESOURCE_MANAGER (application);
+
+ switch (tag) {
+ case TAG_PROFILE_ENQUIRY:
+ ret = send_profile_reply (mgr, session);
+ break;
+ case TAG_PROFILE_REPLY:
+ ret = handle_profile_reply (mgr, session, buffer, length);
+ break;
+ case TAG_PROFILE_CHANGE:
+ ret = send_profile_enquiry (mgr, session);
+ break;
+ default:
+ g_return_val_if_reached (CAM_RETURN_APPLICATION_ERROR);
+ }
+
+ return CAM_RETURN_OK;
+}
diff --git a/sys/dvb/camresourcemanager.h b/sys/dvb/camresourcemanager.h
new file mode 100644
index 00000000..b6fd0a47
--- /dev/null
+++ b/sys/dvb/camresourcemanager.h
@@ -0,0 +1,41 @@
+/*
+ * camresourcemanager.h - GStreamer CAM (EN50221) Resource Manager
+ * Copyright (C) 2007 Alessandro Decina
+ *
+ * Authors:
+ * Alessandro Decina <alessandro@nnva.org>
+ *
+ * This library is free software; you can redistribute it and/or
+ * modify it under the terms of the GNU Library General Public
+ * License as published by the Free Software Foundation; either
+ * version 2 of the License, or (at your option) any later version.
+ *
+ * This library 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
+ * Library General Public License for more details.
+ *
+ * You should have received a copy of the GNU Library General Public
+ * License along with this library; if not, write to the
+ * Free Software Foundation, Inc., 59 Temple Place - Suite 330,
+ * Boston, MA 02111-1307, USA.
+ */
+
+#ifndef CAM_RESOURCE_MANAGER_H
+#define CAM_RESOURCE_MANAGER_H
+
+#include "camapplication.h"
+
+#define CAM_RESOURCE_MANAGER(obj) ((CamResourceManager *) obj)
+
+typedef struct _CamResourceManager CamResourceManager;
+
+struct _CamResourceManager
+{
+ CamALApplication application;
+};
+
+CamResourceManager *cam_resource_manager_new ();
+void cam_resource_manager_destroy (CamResourceManager *manager);
+
+#endif /* CAM_RESOURCE_MANAGER_H */
diff --git a/sys/dvb/camsession.c b/sys/dvb/camsession.c
new file mode 100644
index 00000000..6a2deb9c
--- /dev/null
+++ b/sys/dvb/camsession.c
@@ -0,0 +1,595 @@
+/*
+ * camsession.c - GStreamer CAM (EN50221) Session Layer
+ * Copyright (C) 2007 Alessandro Decina
+ *
+ * Authors:
+ * Alessandro Decina <alessandro@nnva.org>
+ *
+ * This library is free software; you can redistribute it and/or
+ * modify it under the terms of the GNU Library General Public
+ * License as published by the Free Software Foundation; either
+ * version 2 of the License, or (at your option) any later version.
+ *
+ * This library 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
+ * Library General Public License for more details.
+ *
+ * You should have received a copy of the GNU Library General Public
+ * License along with this library; if not, write to the
+ * Free Software Foundation, Inc., 59 Temple Place - Suite 330,
+ * Boston, MA 02111-1307, USA.
+ */
+
+#include "camsession.h"
+
+#define GST_CAT_DEFAULT cam_debug_cat
+#define I_TAG 0
+#define I_LENGTH_FB 1
+
+#define TAG_SESSION_NUMBER 0x90
+#define TAG_OPEN_SESSION_REQUEST 0x91
+#define TAG_OPEN_SESSION_RESPONSE 0x92
+#define TAG_CREATE_SESSION 0x93
+#define TAG_CREATE_SESSION_RESPONSE 0x94
+#define TAG_CLOSE_SESSION_REQUEST 0x95
+#define TAG_CLOSE_SESSION_RESPONSE 0x96
+
+static CamReturn connection_data_cb (CamTL * tl, CamTLConnection * connection,
+ guint8 * spdu, guint spdu_length);
+
+CamSLSession *
+cam_sl_session_new (CamSL * sl, CamTLConnection * connection,
+ guint16 session_nb, guint resource_id)
+{
+ CamSLSession *session = g_new0 (CamSLSession, 1);
+
+ session->state = CAM_SL_SESSION_STATE_IDLE;
+ session->sl = sl;
+ session->connection = connection;
+ session->session_nb = session_nb;
+ session->resource_id = resource_id;
+
+ return session;
+}
+
+void
+cam_sl_session_destroy (CamSLSession * session)
+{
+ g_free (session);
+}
+
+CamSL *
+cam_sl_new (CamTL * tl)
+{
+ CamSL *sl = g_new0 (CamSL, 1);
+
+ sl->sessions = g_hash_table_new_full (g_direct_hash, g_direct_equal,
+ NULL, (GDestroyNotify) cam_sl_session_destroy);
+
+ tl->user_data = sl;
+ tl->connection_data = connection_data_cb;
+
+ return sl;
+}
+
+void
+cam_sl_destroy (CamSL * sl)
+{
+ g_hash_table_destroy (sl->sessions);
+
+ g_free (sl);
+}
+
+CamReturn
+cam_sl_create_session (CamSL * sl,
+ CamTLConnection * connection, guint resource_id,
+ CamSLSession ** out_session)
+{
+ CamReturn ret;
+ CamSLSession *session = NULL;
+ guint size;
+ guint offset;
+ guint8 *tpdu = NULL;
+ guint8 *spdu;
+ guint16 session_nb;
+
+ /* FIXME: implement session number allocations properly */
+ if (sl->session_ids == G_MAXUINT16)
+ return CAM_RETURN_SESSION_TOO_MANY_SESSIONS;
+
+ session_nb = ++sl->session_ids;
+ session = cam_sl_session_new (sl, connection, session_nb, resource_id);
+
+ /* SPDU layout (8 bytes):
+ * TAG_CREATE_SESSION 1 byte
+ * length_field () 1 byte
+ * resource_id 4 bytes
+ * session_nb 2 bytes
+ */
+
+ /* get TPDU size */
+ cam_tl_calc_buffer_size (sl->tl, 8, &size, &offset);
+
+ tpdu = (guint8 *) g_malloc (size);
+ spdu = tpdu + offset;
+
+ /* SPDU header */
+ /* tag */
+ spdu[0] = TAG_CREATE_SESSION;
+ /* fixed length_field */
+ spdu[1] = 6;
+
+ /* SPDU body */
+ /* resource id */
+ GST_WRITE_UINT32_BE (&spdu[2], resource_id);
+ /* session_nb */
+ GST_WRITE_UINT16_BE (&spdu[6], session_nb);
+
+ /* write the TPDU */
+ ret = cam_tl_connection_write (session->connection, tpdu, size, 8);
+ if (CAM_FAILED (ret))
+ goto error;
+
+ *out_session = session;
+
+ g_free (tpdu);
+ return CAM_RETURN_OK;
+
+error:
+ if (session)
+ cam_sl_session_destroy (session);
+
+ g_free (tpdu);
+
+ return ret;
+}
+
+/* send a TAG_CLOSE_SESSION SPDU */
+CamReturn
+cam_sl_session_close (CamSLSession * session)
+{
+ CamReturn ret;
+ guint size;
+ guint offset;
+ guint8 *tpdu = NULL;
+ guint8 *spdu;
+ CamSL *sl = session->sl;
+
+ /* SPDU layout (4 bytes):
+ * TAG_CLOSE_SESSION 1 byte
+ * length_field () 1 byte
+ * session_nb 2 bytes
+ */
+
+ /* get the size of the TPDU */
+ cam_tl_calc_buffer_size (sl->tl, 4, &size, &offset);
+
+ tpdu = (guint8 *) g_malloc (size);
+ /* the spdu header starts after the TPDU headers */
+ spdu = tpdu + offset;
+
+ /* SPDU header */
+ /* tag */
+ spdu[0] = TAG_CLOSE_SESSION_REQUEST;
+ /* fixed length_field */
+ spdu[1] = 2;
+ /* SPDU body */
+ /* session_nb */
+ GST_WRITE_UINT16_BE (&spdu[2], session->session_nb);
+
+ /* write the TPDU */
+ ret = cam_tl_connection_write (session->connection, tpdu, size, 4);
+ if (CAM_FAILED (ret))
+ goto error;
+
+ session->state = CAM_SL_SESSION_STATE_CLOSING;
+
+ g_free (tpdu);
+
+ return CAM_RETURN_OK;
+
+error:
+ g_free (tpdu);
+
+ return ret;
+}
+
+void
+cam_sl_calc_buffer_size (CamSL * sl, guint body_length,
+ guint * buffer_size, guint * offset)
+{
+ /* an APDU is sent in a SESSION_NUMBER SPDU, which has a fixed header size (4
+ * bytes) */
+ cam_tl_calc_buffer_size (sl->tl, 4 + body_length, buffer_size, offset);
+ *offset += 4;
+}
+
+CamReturn
+cam_sl_session_write (CamSLSession * session,
+ guint8 * buffer, guint buffer_size, guint body_length)
+{
+ guint8 *spdu;
+
+ /* SPDU layout (4 + body_length bytes):
+ * TAG_SESSION_NUMBER (1 byte)
+ * length_field (1 byte)
+ * session number (2 bytes)
+ * one or more APDUs (body_length bytes)
+ */
+
+ spdu = (buffer + buffer_size) - body_length - 4;
+ spdu[0] = TAG_SESSION_NUMBER;
+ spdu[1] = 2;
+ GST_WRITE_UINT16_BE (&spdu[2], session->session_nb);
+
+ /* add our header to the body length */
+ return cam_tl_connection_write (session->connection,
+ buffer, buffer_size, 4 + body_length);
+}
+
+static CamReturn
+send_open_session_response (CamSL * sl, CamSLSession * session, guint8 status)
+{
+ CamReturn ret;
+ guint8 *tpdu;
+ guint size;
+ guint offset;
+ guint8 *spdu;
+
+ /* SPDU layout (9 bytes):
+ * TAG_OPEN_SESSION_RESPONSE 1 byte
+ * length_field () 1 byte
+ * session_status 1 byte
+ * resource_id 4 bytes
+ * session_nb 2 bytes
+ */
+
+ cam_tl_calc_buffer_size (session->sl->tl, 9, &size, &offset);
+
+ tpdu = g_malloc0 (size);
+ spdu = tpdu + offset;
+
+ spdu[0] = TAG_OPEN_SESSION_RESPONSE;
+ /* fixed length_field () */
+ spdu[1] = 7;
+ spdu[2] = status;
+ GST_WRITE_UINT32_BE (&spdu[3], session->resource_id);
+ GST_WRITE_UINT16_BE (&spdu[7], session->session_nb);
+
+ ret = cam_tl_connection_write (session->connection, tpdu, size, 9);
+ g_free (tpdu);
+ if (CAM_FAILED (ret))
+ return ret;
+
+ return CAM_RETURN_OK;
+}
+
+static CamReturn
+send_close_session_response (CamSL * sl, CamSLSession * session, guint8 status)
+{
+ CamReturn ret;
+ guint8 *tpdu;
+ guint size;
+ guint offset;
+ guint8 *spdu;
+
+ /* SPDU layout (5 bytes):
+ * TAG_CLOSE_SESSION_RESPONSE 1 byte
+ * length_field () 1 byte
+ * session_status 1 byte
+ * session_nb 2 bytes
+ */
+
+ cam_tl_calc_buffer_size (session->sl->tl, 5, &size, &offset);
+
+ tpdu = g_malloc0 (size);
+ spdu = tpdu + offset;
+
+ spdu[0] = TAG_OPEN_SESSION_RESPONSE;
+ /* fixed length_field() */
+ spdu[1] = 3;
+ spdu[2] = status;
+ GST_WRITE_UINT16_BE (&spdu[3], session->session_nb);
+
+ ret = cam_tl_connection_write (session->connection, tpdu, size, 5);
+ g_free (tpdu);
+ if (CAM_FAILED (ret))
+ return ret;
+
+ return CAM_RETURN_OK;
+}
+
+static CamReturn
+handle_open_session_request (CamSL * sl, CamTLConnection * connection,
+ guint8 * spdu, guint spdu_length)
+{
+ CamReturn ret;
+ guint resource_id;
+ guint status;
+ guint16 session_nb;
+ CamSLSession *session;
+
+ /* SPDU layout (6 bytes):
+ * TAG_OPEN_SESSION_REQUEST (1 byte)
+ * length_field() (1 byte)
+ * resource id (4 bytes)
+ */
+ if (spdu_length != 6) {
+ GST_ERROR ("expected OPEN_SESSION_REQUEST to be 6 bytes, got %d",
+ spdu_length);
+ return CAM_RETURN_SESSION_ERROR;
+ }
+
+ /* skip tag and length_field () */
+ resource_id = GST_READ_UINT32_BE (&spdu[2]);
+
+ /* create a new session */
+ if (sl->session_ids == G_MAXUINT16) {
+ GST_ERROR ("too many sessions opened");
+ return CAM_RETURN_SESSION_TOO_MANY_SESSIONS;
+ }
+
+ session_nb = ++sl->session_ids;
+ session = cam_sl_session_new (sl, connection, session_nb, resource_id);
+
+ GST_INFO ("session request: %d %x", session_nb, session->resource_id);
+
+ if (sl->open_session_request) {
+ /* forward the request to the upper layer */
+ ret = sl->open_session_request (sl, session, &status);
+ if (CAM_FAILED (ret))
+ goto error;
+ } else {
+ status = 0xF0;
+ }
+
+ ret = send_open_session_response (sl, session, (guint8) status);
+ if (CAM_FAILED (ret))
+ goto error;
+
+ GST_INFO ("session request response: %d %x", session_nb, status);
+
+ if (status == CAM_SL_RESOURCE_STATUS_OPEN) {
+ /* if the session has been accepted add it and signal */
+ session->state = CAM_SL_SESSION_STATE_ACTIVE;
+ g_hash_table_insert (sl->sessions,
+ GINT_TO_POINTER ((guint) session_nb), session);
+
+ if (sl->session_opened) {
+ /* notify the upper layer */
+ ret = sl->session_opened (sl, session);
+ if (CAM_FAILED (ret))
+ return ret;
+ }
+ } else {
+ /* session request wasn't accepted */
+ cam_sl_session_destroy (session);
+ }
+
+ return CAM_RETURN_OK;
+
+error:
+ cam_sl_session_destroy (session);
+
+ return ret;
+}
+
+static CamReturn
+handle_create_session_response (CamSL * sl, CamTLConnection * connection,
+ guint8 * spdu, guint spdu_length)
+{
+ guint8 status;
+ guint resource_id;
+ guint16 session_nb;
+ CamSLSession *session;
+
+ /* SPDU layout (9 bytes):
+ * TAG_CREATE_SESSION_RESPONSE (1 byte)
+ * length_field() (1 byte)
+ * status (1 byte)
+ * resource id (4 bytes)
+ * session number (2 bytes)
+ */
+ if (spdu_length != 9) {
+ GST_ERROR ("expected CREATE_SESSION_RESPONSE to be 9 bytes, got %d",
+ spdu_length);
+ return CAM_RETURN_SESSION_ERROR;
+ }
+
+ /* skip tag and length */
+ status = spdu[2];
+ resource_id = GST_READ_UINT32_BE (&spdu[3]);
+ session_nb = GST_READ_UINT16_BE (&spdu[7]);
+
+ session = g_hash_table_lookup (sl->sessions,
+ GINT_TO_POINTER ((guint) session_nb));
+ if (session == NULL) {
+ GST_DEBUG ("got CREATE_SESSION_RESPONSE for unknown session: %d",
+ session_nb);
+ return CAM_RETURN_SESSION_ERROR;
+ }
+
+ if (session->state == CAM_SL_SESSION_STATE_CLOSING) {
+ GST_DEBUG ("ignoring CREATE_SESSION_RESPONSE for closing session: %d",
+ session_nb);
+ return CAM_RETURN_OK;
+ }
+
+ session->state = CAM_SL_SESSION_STATE_ACTIVE;
+
+ GST_DEBUG ("session opened %d", session->session_nb);
+
+ if (sl->session_opened)
+ /* notify the upper layer */
+ return sl->session_opened (sl, session);
+ return CAM_RETURN_OK;
+}
+
+static CamReturn
+handle_close_session_request (CamSL * sl, CamTLConnection * connection,
+ guint8 * spdu, guint spdu_length)
+{
+ CamReturn ret;
+ guint16 session_nb;
+ CamSLSession *session;
+ guint8 status = 0;
+
+ /* SPDU layout (4 bytes):
+ * TAG_CLOSE_SESSION_REQUEST (1 byte)
+ * length_field () (1 byte)
+ * session number (2 bytes)
+ */
+ if (spdu_length != 4) {
+ GST_ERROR ("expected CLOSE_SESSION_REQUEST to be 4 bytes, got %d",
+ spdu_length);
+ return CAM_RETURN_SESSION_ERROR;
+ }
+
+ /* skip tag and length_field() */
+ session_nb = GST_READ_UINT16_BE (&spdu[2]);
+
+ GST_DEBUG ("close session request %d", session_nb);
+
+ session = g_hash_table_lookup (sl->sessions,
+ GINT_TO_POINTER ((guint) session_nb));
+ if (session == NULL) {
+ GST_WARNING ("got CLOSE_SESSION_REQUEST for unknown session: %d",
+ session_nb);
+
+ status = 0xF0;
+ } else if (session->state == CAM_SL_SESSION_STATE_CLOSING) {
+ GST_WARNING ("got CLOSE_SESSION_REQUEST for closing session: %d",
+ session_nb);
+
+ status = 0xF0;
+ }
+
+ GST_DEBUG ("close session response: %d %d", session->session_nb, status);
+
+ ret = send_close_session_response (sl, session, status);
+ if (CAM_FAILED (ret))
+ return ret;
+
+ if (session->state != CAM_SL_SESSION_STATE_CLOSING) {
+ GST_DEBUG ("session closed %d", session->session_nb);
+
+ if (sl->session_closed)
+ ret = sl->session_closed (sl, session);
+
+ g_hash_table_remove (sl->sessions,
+ GINT_TO_POINTER ((guint) session->session_nb));
+
+ if (CAM_FAILED (ret))
+ return ret;
+ }
+
+ return CAM_RETURN_OK;
+}
+
+static CamReturn
+handle_close_session_response (CamSL * sl, CamTLConnection * connection,
+ guint8 * spdu, guint spdu_length)
+{
+ guint16 session_nb;
+ CamSLSession *session;
+ CamReturn ret = CAM_RETURN_OK;
+
+ /* SPDU layout (5 bytes):
+ * TAG_CLOSE_SESSION_RESPONSE (1 byte)
+ * length_field () (1 byte)
+ * status (1 byte)
+ * session number (2 bytes)
+ */
+
+ if (spdu_length != 5) {
+ GST_ERROR ("expected CLOSE_SESSION_RESPONSE to be 5 bytes, got %d",
+ spdu_length);
+ return CAM_RETURN_SESSION_ERROR;
+ }
+
+ /* skip tag, length_field() and session_status */
+ session_nb = GST_READ_UINT16_BE (&spdu[3]);
+
+ session = g_hash_table_lookup (sl->sessions,
+ GINT_TO_POINTER ((guint) session_nb));
+ if (session == NULL || session->state != CAM_SL_SESSION_STATE_ACTIVE) {
+ GST_ERROR ("unexpected CLOSED_SESSION_RESPONSE");
+ return CAM_RETURN_SESSION_ERROR;
+ }
+
+ GST_DEBUG ("session closed %d", session->session_nb);
+
+ if (sl->session_closed)
+ ret = sl->session_closed (sl, session);
+
+ g_hash_table_remove (sl->sessions,
+ GINT_TO_POINTER ((guint) session->session_nb));
+
+ return ret;
+}
+
+static CamReturn
+handle_session_data (CamSL * sl, CamTLConnection * connection,
+ guint8 * spdu, guint length)
+{
+ guint16 session_nb;
+ CamSLSession *session;
+
+ /* SPDU layout (>= 4 bytes):
+ * TAG_SESSION_NUMBER (1 byte)
+ * length_field() (1 byte)
+ * session number (2 bytes)
+ * one or more APDUs
+ */
+
+ if (length < 4) {
+ GST_ERROR ("invalid SESSION_NUMBER SPDU length %d", length);
+ return CAM_RETURN_SESSION_ERROR;
+ }
+
+ session_nb = GST_READ_UINT16_BE (&spdu[2]);
+
+ session = g_hash_table_lookup (sl->sessions,
+ GINT_TO_POINTER ((guint) session_nb));
+ if (session == NULL) {
+ GST_ERROR ("got SESSION_NUMBER on an unknown connection: %d", session_nb);
+ return CAM_RETURN_SESSION_ERROR;
+ }
+
+ if (sl->session_data)
+ /* pass the APDUs to the upper layer, removing our 4-bytes header */
+ return sl->session_data (sl, session, spdu + 4, length - 4);
+
+ return CAM_RETURN_OK;
+}
+
+static CamReturn
+connection_data_cb (CamTL * tl, CamTLConnection * connection,
+ guint8 * spdu, guint spdu_length)
+{
+ CamReturn ret;
+ CamSL *sl = CAM_SL (tl->user_data);
+
+ switch (spdu[I_TAG]) {
+ case TAG_CREATE_SESSION_RESPONSE:
+ ret = handle_create_session_response (sl, connection, spdu, spdu_length);
+ break;
+ case TAG_OPEN_SESSION_REQUEST:
+ ret = handle_open_session_request (sl, connection, spdu, spdu_length);
+ break;
+ case TAG_CLOSE_SESSION_REQUEST:
+ ret = handle_close_session_request (sl, connection, spdu, spdu_length);
+ break;
+ case TAG_CLOSE_SESSION_RESPONSE:
+ ret = handle_close_session_response (sl, connection, spdu, spdu_length);
+ break;
+ case TAG_SESSION_NUMBER:
+ ret = handle_session_data (sl, connection, spdu, spdu_length);
+ break;
+ default:
+ g_return_val_if_reached (CAM_RETURN_SESSION_ERROR);
+ }
+
+ return ret;
+}
diff --git a/sys/dvb/camsession.h b/sys/dvb/camsession.h
new file mode 100644
index 00000000..7b9ac54b
--- /dev/null
+++ b/sys/dvb/camsession.h
@@ -0,0 +1,96 @@
+/*
+ * camsession.h - GStreamer CAM (EN50221) Session Layer
+ * Copyright (C) 2007 Alessandro Decina
+ *
+ * Authors:
+ * Alessandro Decina <alessandro@nnva.org>
+ *
+ * This library is free software; you can redistribute it and/or
+ * modify it under the terms of the GNU Library General Public
+ * License as published by the Free Software Foundation; either
+ * version 2 of the License, or (at your option) any later version.
+ *
+ * This library 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
+ * Library General Public License for more details.
+ *
+ * You should have received a copy of the GNU Library General Public
+ * License along with this library; if not, write to the
+ * Free Software Foundation, Inc., 59 Temple Place - Suite 330,
+ * Boston, MA 02111-1307, USA.
+ */
+
+#ifndef CAM_SESSION_H
+#define CAM_SESSION_H
+
+#include "cam.h"
+#include "camtransport.h"
+
+#define CAM_SL(obj) ((CamSL *) obj)
+#define CAM_SL_SESSION(obj) ((CamSLSession *) obj)
+
+typedef struct _CamSL CamSL;
+typedef struct _CamSLSession CamSLSession;
+
+typedef enum
+{
+ CAM_SL_SESSION_STATE_IDLE,
+ CAM_SL_SESSION_STATE_OPENING,
+ CAM_SL_SESSION_STATE_ACTIVE,
+ CAM_SL_SESSION_STATE_CLOSING
+} CamSLSessionState;
+
+typedef enum
+{
+ CAM_SL_RESOURCE_STATUS_OPEN = 0x00,
+ CAM_SL_RESOURCE_STATUS_NOT_FOUND = 0xF0,
+ CAM_SL_RESOURCE_STATUS_UNAVAILABLE = 0xF1,
+ CAM_SL_RESOURCE_STATUS_INVALID_VERSION = 0xF2,
+ CAM_SL_RESOURCE_STATUS_BUSY = 0xF3
+} CamSLResourceStatus;
+
+struct _CamSL
+{
+ CamTL *tl;
+
+ GHashTable *sessions;
+ guint session_ids;
+
+ /* callbacks */
+ CamReturn (*open_session_request) (CamSL *sl, CamSLSession *session,
+ CamSLResourceStatus *status);
+ CamReturn (*session_opened) (CamSL *sl, CamSLSession *session);
+ CamReturn (*session_closed) (CamSL *sl, CamSLSession *session);
+ CamReturn (*session_data) (CamSL *sl, CamSLSession *session,
+ guint8 *data, guint length);
+
+ gpointer user_data;
+};
+
+struct _CamSLSession
+{
+ CamSL *sl;
+ CamTLConnection *connection;
+
+ guint resource_id;
+ guint16 session_nb;
+
+ CamSLSessionState state;
+
+ gpointer user_data;
+};
+
+CamSL *cam_sl_new (CamTL *transport);
+void cam_sl_destroy (CamSL *sl);
+
+CamReturn cam_sl_create_session (CamSL *sl, CamTLConnection *connection,
+ guint resource_id, CamSLSession **session);
+CamReturn cam_sl_session_close (CamSLSession *session);
+
+void cam_sl_calc_buffer_size (CamSL *sl,
+ guint body_length, guint *buffer_size, guint *offset);
+CamReturn cam_sl_session_write (CamSLSession *session,
+ guint8 *buffe, guint buffer_size, guint body_length);
+
+#endif /* CAM_SESSION_H */
diff --git a/sys/dvb/camswclient.c b/sys/dvb/camswclient.c
new file mode 100644
index 00000000..d956f3c9
--- /dev/null
+++ b/sys/dvb/camswclient.c
@@ -0,0 +1,162 @@
+/*
+ * camswclient.c - GStreamer softcam client
+ * Copyright (C) 2007 Alessandro Decina
+ *
+ * Authors:
+ * Alessandro Decina <alessandro@nnva.org>
+ *
+ * This library is free software; you can redistribute it and/or
+ * modify it under the terms of the GNU Library General Public
+ * License as published by the Free Software Foundation; either
+ * version 2 of the License, or (at your option) any later version.
+ *
+ * This library 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
+ * Library General Public License for more details.
+ *
+ * You should have received a copy of the GNU Library General Public
+ * License along with this library; if not, write to the
+ * Free Software Foundation, Inc., 59 Temple Place - Suite 330,
+ * Boston, MA 02111-1307, USA.
+ */
+
+#include <sys/types.h>
+#include <sys/socket.h>
+#include <sys/un.h>
+#include <unistd.h>
+
+#include <gst/gst.h>
+
+#include "camswclient.h"
+#include "cam.h"
+#include "camutils.h"
+
+#define GST_CAT_DEFAULT cam_debug_cat
+#define UNIX_PATH_MAX 108
+
+CamSwClient *
+cam_sw_client_new ()
+{
+ CamSwClient *client = g_new0 (CamSwClient, 1);
+
+ client->state = CAM_SW_CLIENT_STATE_CLOSED;
+
+ return client;
+}
+
+static void
+reset_state (CamSwClient * client)
+{
+ if (client->sock)
+ close (client->sock);
+
+ if (client->sock_path)
+ g_free (client->sock_path);
+}
+
+void
+cam_sw_client_free (CamSwClient * client)
+{
+ g_return_if_fail (client != NULL);
+
+ if (client->state != CAM_SW_CLIENT_STATE_CLOSED)
+ GST_WARNING ("client not in CLOSED state when free'd");
+
+ reset_state (client);
+ g_free (client);
+}
+
+gboolean
+cam_sw_client_open (CamSwClient * client, const char *sock_path)
+{
+ struct sockaddr_un addr;
+ int ret;
+
+ g_return_val_if_fail (client != NULL, FALSE);
+ g_return_val_if_fail (client->state == CAM_SW_CLIENT_STATE_CLOSED, FALSE);
+ g_return_val_if_fail (sock_path != NULL, FALSE);
+
+ addr.sun_family = AF_UNIX;
+ strncpy (addr.sun_path, sock_path, UNIX_PATH_MAX);
+
+ GST_INFO ("connecting to softcam socket: %s", sock_path);
+ client->sock = socket (PF_UNIX, SOCK_STREAM, 0);
+ ret =
+ connect (client->sock, (struct sockaddr *) &addr,
+ sizeof (struct sockaddr_un));
+ if (ret != 0) {
+ GST_ERROR ("error opening softcam socket %s, error: %s",
+ sock_path, strerror (errno));
+
+ return FALSE;
+ }
+
+ client->sock_path = g_strdup (sock_path);
+ client->state = CAM_SW_CLIENT_STATE_OPEN;
+
+ return TRUE;
+}
+
+void
+cam_sw_client_close (CamSwClient * client)
+{
+ g_return_if_fail (client != NULL);
+ g_return_if_fail (client->state == CAM_SW_CLIENT_STATE_OPEN);
+
+ reset_state (client);
+ client->state = CAM_SW_CLIENT_STATE_CLOSED;
+}
+
+static void
+send_ca_pmt (CamSwClient * client, GObject * pmt,
+ guint8 list_management, guint8 cmd_id)
+{
+ guint8 *buffer;
+ guint buffer_size;
+ guint8 *ca_pmt;
+ guint ca_pmt_size;
+ guint length_field_len;
+ guint header_len;
+
+ ca_pmt = cam_build_ca_pmt (pmt, list_management, cmd_id, &ca_pmt_size);
+
+ length_field_len = cam_calc_length_field_size (ca_pmt_size);
+ header_len = 3 + length_field_len;
+ buffer_size = header_len + ca_pmt_size;
+
+ buffer = g_malloc0 (buffer_size);
+ memcpy (buffer + header_len, ca_pmt, ca_pmt_size);
+
+ /* ca_pmt resource_id */
+ buffer[0] = 0x9F;
+ buffer[1] = 0x80;
+ buffer[2] = 0x32;
+
+ cam_write_length_field (&buffer[3], ca_pmt_size);
+
+ write (client->sock, buffer, buffer_size);
+
+ g_free (ca_pmt);
+ g_free (buffer);
+}
+
+void
+cam_sw_client_set_pmt (CamSwClient * client, GObject * pmt)
+{
+ g_return_if_fail (client != NULL);
+ g_return_if_fail (pmt != NULL);
+
+ return send_ca_pmt (client, pmt, 0x03 /* only */ ,
+ 0x01 /* ok_descrambling */ );
+}
+
+void
+cam_sw_client_update_pmt (CamSwClient * client, GObject * pmt)
+{
+ g_return_if_fail (client != NULL);
+ g_return_if_fail (pmt != NULL);
+
+ return send_ca_pmt (client, pmt, 0x05 /* update */ ,
+ 0x01 /* ok_descrambling */ );
+}
diff --git a/sys/dvb/camswclient.h b/sys/dvb/camswclient.h
new file mode 100644
index 00000000..5471e96a
--- /dev/null
+++ b/sys/dvb/camswclient.h
@@ -0,0 +1,53 @@
+/*
+ * camswclient.h - GStreamer softcam client
+ * Copyright (C) 2007 Alessandro Decina
+ *
+ * Authors:
+ * Alessandro Decina <alessandro@nnva.org>
+ *
+ * This library is free software; you can redistribute it and/or
+ * modify it under the terms of the GNU Library General Public
+ * License as published by the Free Software Foundation; either
+ * version 2 of the License, or (at your option) any later version.
+ *
+ * This library 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
+ * Library General Public License for more details.
+ *
+ * You should have received a copy of the GNU Library General Public
+ * License along with this library; if not, write to the
+ * Free Software Foundation, Inc., 59 Temple Place - Suite 330,
+ * Boston, MA 02111-1307, USA.
+ */
+
+#ifndef CAM_SW_CLIENT_H
+#define CAM_SW_CLIENT_H
+
+#include <glib.h>
+
+typedef enum
+{
+ CAM_SW_CLIENT_STATE_CLOSED,
+ CAM_SW_CLIENT_STATE_OPEN,
+} CamSwClientState;
+
+typedef struct
+{
+ /* private */
+ CamSwClientState state;
+ char *sock_path;
+ int sock;
+
+} CamSwClient;
+
+CamSwClient *cam_sw_client_new ();
+void cam_sw_client_free (CamSwClient *sw_client);
+
+gboolean cam_sw_client_open (CamSwClient *sw_client, const char *sock_path);
+void cam_sw_client_close (CamSwClient *sw_client);
+
+void cam_sw_client_set_pmt (CamSwClient *sw_client, GObject *pmt);
+void cam_sw_client_update_pmt (CamSwClient *sw_client, GObject *pmt);
+
+#endif /* CAM_SW_CLIENT_H */
diff --git a/sys/dvb/camtransport.c b/sys/dvb/camtransport.c
new file mode 100644
index 00000000..288b2e46
--- /dev/null
+++ b/sys/dvb/camtransport.c
@@ -0,0 +1,505 @@
+/*
+ * camtransport.c - GStreamer CAM (EN50221) transport layer
+ * Copyright (C) 2007 Alessandro Decina
+ *
+ * Authors:
+ * Alessandro Decina <alessandro@nnva.org>
+ *
+ * This library is free software; you can redistribute it and/or
+ * modify it under the terms of the GNU Library General Public
+ * License as published by the Free Software Foundation; either
+ * version 2 of the License, or (at your option) any later version.
+ *
+ * This library 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
+ * Library General Public License for more details.
+ *
+ * You should have received a copy of the GNU Library General Public
+ * License along with this library; if not, write to the
+ * Free Software Foundation, Inc., 59 Temple Place - Suite 330,
+ * Boston, MA 02111-1307, USA.
+ */
+
+#include "camtransport.h"
+#include <sys/select.h>
+#include <sys/time.h>
+#include <sys/types.h>
+#include <unistd.h>
+#include <errno.h>
+
+#define GST_CAT_DEFAULT cam_debug_cat
+#define READ_TIMEOUT_SEC 2
+#define READ_TIMEOUT_USEC 0
+
+#define POLL_INTERVAL 0.300
+
+#define TAG_SB 0x80
+#define TAG_RCV 0x81
+#define TAG_CREATE_T_C 0x82
+#define TAG_C_T_C_REPLY 0x83
+#define TAG_DELETE_T_C 0x84
+#define TAG_D_T_C_REPLY 0x85
+#define TAG_REQUEST_T_C 0x86
+#define TAG_NEW_T_C 0x87
+#define TAG_T_C_ERROR 0x88
+#define TAG_DATA_MORE 0xA1
+#define TAG_DATA_LAST 0xA0
+
+/* utility struct used to store the state of the connections in cam_tl_read_next
+ */
+typedef struct
+{
+ GList *active;
+ GList *idle;
+} CamTLConnectionsStatus;
+
+void cam_gst_util_dump_mem (const guchar * mem, guint size);
+
+CamTLConnection *
+cam_tl_connection_new (CamTL * tl, guint8 id)
+{
+ CamTLConnection *connection;
+
+ connection = g_new0 (CamTLConnection, 1);
+ connection->tl = tl;
+ connection->id = id;
+ connection->state = CAM_TL_CONNECTION_STATE_CLOSED;
+ connection->has_data = FALSE;
+
+ return connection;
+}
+
+static void
+cam_tl_connection_destroy (CamTLConnection * connection)
+{
+ if (connection->last_poll)
+ g_timer_destroy (connection->last_poll);
+
+ g_free (connection);
+}
+
+CamTL *
+cam_tl_new (int fd)
+{
+ CamTL *tl;
+
+ tl = g_new0 (CamTL, 1);
+ tl->fd = fd;
+ tl->connections = g_hash_table_new_full (g_direct_hash, g_direct_equal,
+ NULL, (GDestroyNotify) cam_tl_connection_destroy);
+
+ return tl;
+}
+
+void
+cam_tl_destroy (CamTL * tl)
+{
+ g_hash_table_destroy (tl->connections);
+
+ g_free (tl);
+}
+
+/* read data from the module without blocking indefinitely */
+static CamReturn
+cam_tl_read_timeout (CamTL * tl, struct timeval *timeout)
+{
+ fd_set read_fd;
+ int sret;
+
+ FD_ZERO (&read_fd);
+ FD_SET (tl->fd, &read_fd);
+
+ sret = select (tl->fd + 1, &read_fd, NULL, NULL, timeout);
+ if (sret == 0) {
+ GST_DEBUG ("read timeout");
+ return CAM_RETURN_TRANSPORT_TIMEOUT;
+ }
+
+ tl->buffer_size = read (tl->fd, &tl->buffer, HOST_BUFFER_SIZE);
+ if (tl->buffer_size == -1) {
+ GST_ERROR ("error reading tpdu: %s", g_strerror (errno));
+ return CAM_RETURN_TRANSPORT_ERROR;
+ }
+
+ return CAM_RETURN_OK;
+}
+
+/* read data from the module using the default timeout */
+static CamReturn
+cam_tl_read (CamTL * tl)
+{
+ struct timeval timeout;
+
+ timeout.tv_sec = READ_TIMEOUT_SEC;
+ timeout.tv_usec = READ_TIMEOUT_USEC;
+
+ return cam_tl_read_timeout (tl, &timeout);
+}
+
+/* get the number of bytes to allocate for a TPDU with a body of body_length
+ * bytes. Also get the offset from the beginning of the buffer that marks the
+ * position of the first byte of the TPDU body */
+void
+cam_tl_calc_buffer_size (CamTL * tl, guint body_length,
+ guint * buffer_size, guint * offset)
+{
+ guint length_field_len;
+
+ /* the size of a TPDU is:
+ * 1 byte slot number
+ * 1 byte connection id
+ * length_field_len bytes length field
+ * 1 byte connection id
+ * body_length bytes body
+ */
+
+ /* get the length of the lenght_field block */
+ length_field_len = cam_calc_length_field_size (body_length);
+
+ *offset = 3 + length_field_len + 1;
+ *buffer_size = *offset + body_length;
+}
+
+/* write the header of a TPDU
+ * NOTE: this function assumes that the buffer is large enough to contain the
+ * complete TPDU (see cam_tl_calc_buffer_size ()) and that enough space has been
+ * left from the beginning of the buffer to write the TPDU header.
+ */
+static CamReturn
+cam_tl_connection_write_tpdu (CamTLConnection * connection,
+ guint8 tag, guint8 * buffer, guint buffer_size, guint body_length)
+{
+ int sret;
+ CamTL *tl = connection->tl;
+ guint8 length_field_len;
+
+ /* slot number */
+ buffer[0] = connection->slot;
+ /* connection number */
+ buffer[1] = connection->id;
+ /* tag */
+ buffer[2] = tag;
+ /* length can take 1 to 4 bytes */
+ length_field_len = cam_write_length_field (&buffer[3], body_length);
+ buffer[3 + length_field_len] = connection->id;
+
+ GST_DEBUG ("writing TPDU %x connection %d", buffer[2], connection->id);
+
+ //cam_gst_util_dump_mem (buffer, buffer_size);
+
+ sret = write (tl->fd, buffer, buffer_size);
+ if (sret == -1) {
+ GST_ERROR ("error witing TPDU (%d): %s", errno, g_strerror (errno));
+ return CAM_RETURN_TRANSPORT_ERROR;
+ }
+
+ tl->expected_tpdus += 1;
+
+ return CAM_RETURN_OK;
+}
+
+/* convenience function to write control TPDUs (TPDUs having a single-byte body)
+ */
+static CamReturn
+cam_tl_connection_write_control_tpdu (CamTLConnection * connection, guint8 tag)
+{
+ guint8 tpdu[5];
+
+ /* TPDU layout (5 bytes):
+ *
+ * slot number (1 byte)
+ * connection id (1 byte)
+ * tag (1 byte)
+ * length (1 byte)
+ * connection id (1 byte)
+ */
+
+ return cam_tl_connection_write_tpdu (connection, tag, tpdu, 5, 1);
+}
+
+/* read the next TPDU from the CAM */
+static CamReturn
+cam_tl_read_tpdu_next (CamTL * tl, CamTLConnection ** out_connection)
+{
+ CamReturn ret;
+ CamTLConnection *connection;
+ guint8 slot;
+ guint8 connection_id;
+ guint8 *tpdu;
+ guint8 length_field_len;
+ guint8 status;
+
+ ret = cam_tl_read (tl);
+ if (CAM_FAILED (ret))
+ return ret;
+
+ tpdu = tl->buffer;
+
+ /* must hold at least slot, connection_id, 1byte length_field, connection_id
+ */
+ if (tl->buffer_size < 4) {
+ GST_ERROR ("invalid TPDU length %d", tl->buffer_size);
+ return CAM_RETURN_TRANSPORT_ERROR;
+ }
+
+ /* LPDU slot */
+ slot = tpdu[0];
+ /* LPDU connection id */
+ connection_id = tpdu[1];
+
+ connection = g_hash_table_lookup (tl->connections,
+ GINT_TO_POINTER ((guint) connection_id));
+ if (connection == NULL) {
+ /* WHAT? */
+ GST_ERROR ("CAM sent a TPDU on an unknown connection: %d", connection_id);
+ return CAM_RETURN_TRANSPORT_ERROR;
+ }
+
+ /* read the length_field () */
+ length_field_len = cam_read_length_field (&tpdu[3], &tl->body_length);
+
+ if (tl->body_length + 3 > tl->buffer_size) {
+ GST_ERROR ("invalid TPDU length_field (%d) exceeds "
+ "the size of the buffer (%d)", tl->body_length, tl->buffer_size);
+ return CAM_RETURN_TRANSPORT_ERROR;
+ }
+
+ /* skip slot + connection id + tag + lenght_field () + connection id */
+ tl->body = tpdu + 4 + length_field_len;
+ /* do not count the connection id byte as part of the body */
+ tl->body_length -= 1;
+
+ if (tl->buffer[tl->buffer_size - 4] != TAG_SB) {
+ GST_ERROR ("no TAG_SB appended to TPDU");
+ return CAM_RETURN_TRANSPORT_ERROR;
+ }
+
+ status = tl->buffer[tl->buffer_size - 1];
+ if (status & 0x80) {
+ connection->has_data = TRUE;
+ } else {
+ connection->has_data = FALSE;
+ }
+
+ GST_DEBUG ("received TPDU %x more data %d", tpdu[2], connection->has_data);
+ tl->expected_tpdus -= 1;
+
+ *out_connection = connection;
+
+ return CAM_RETURN_OK;
+}
+
+/* create a connection with the module */
+CamReturn
+cam_tl_create_connection (CamTL * tl, guint8 slot,
+ CamTLConnection ** connection)
+{
+ CamReturn ret;
+ CamTLConnection *conn = NULL;
+
+ if (tl->connection_ids == 255)
+ return CAM_RETURN_TRANSPORT_TOO_MANY_CONNECTIONS;
+
+ conn = cam_tl_connection_new (tl, ++tl->connection_ids);
+
+ /* send a TAG_CREATE_T_C TPDU */
+ ret = cam_tl_connection_write_control_tpdu (conn, TAG_CREATE_T_C);
+ if (CAM_FAILED (ret))
+ goto error;
+
+ g_hash_table_insert (tl->connections, GINT_TO_POINTER (conn->id), conn);
+
+ *connection = conn;
+
+ return CAM_RETURN_OK;
+
+error:
+ if (conn)
+ cam_tl_connection_destroy (conn);
+
+ return ret;
+}
+
+CamReturn
+cam_tl_connection_delete (CamTLConnection * connection)
+{
+ CamReturn ret;
+
+ ret = cam_tl_connection_write_control_tpdu (connection, TAG_DELETE_T_C);
+ if (CAM_FAILED (ret))
+ return ret;
+
+ connection->state = CAM_TL_CONNECTION_STATE_IN_DELETION;
+
+ return CAM_RETURN_OK;
+}
+
+CamReturn
+handle_control_tpdu (CamTL * tl, CamTLConnection * connection)
+{
+ if (tl->body_length != 0) {
+ GST_ERROR ("got control tpdu of invalid length: %d", tl->body_length);
+ return CAM_RETURN_TRANSPORT_ERROR;
+ }
+
+ switch (tl->buffer[2]) {
+ /* create transport connection reply */
+ case TAG_C_T_C_REPLY:
+ /* a connection might be closed before it's acknowledged */
+ if (connection->state != CAM_TL_CONNECTION_STATE_IN_DELETION) {
+ GST_DEBUG ("connection created %d", connection->id);
+ connection->state = CAM_TL_CONNECTION_STATE_OPEN;
+
+ if (tl->connection_created)
+ tl->connection_created (tl, connection);
+ }
+ break;
+ /* delete transport connection reply */
+ case TAG_D_T_C_REPLY:
+ connection->state = CAM_TL_CONNECTION_STATE_CLOSED;
+ GST_DEBUG ("connection closed %d", connection->id);
+
+ if (tl->connection_deleted)
+ tl->connection_deleted (tl, connection);
+
+ g_hash_table_remove (tl->connections,
+ GINT_TO_POINTER ((guint) connection->id));
+ break;
+ }
+
+ return CAM_RETURN_OK;
+}
+
+CamReturn
+handle_data_tpdu (CamTL * tl, CamTLConnection * connection)
+{
+ if (tl->body_length == 0) {
+ /* FIXME: figure out why this seems to happen from time to time with the
+ * predator cam */
+ GST_WARNING ("Empty data TPDU received");
+ return CAM_RETURN_OK;
+ }
+
+ if (tl->connection_data)
+ return tl->connection_data (tl, connection, tl->body, tl->body_length);
+
+ return CAM_RETURN_OK;
+}
+
+static void
+foreach_connection_get (gpointer key, gpointer value, gpointer user_data)
+{
+ GList **lst = (GList **) user_data;
+
+ *lst = g_list_append (*lst, value);
+}
+
+CamReturn
+cam_tl_connection_poll (CamTLConnection * connection, gboolean force)
+{
+ CamReturn ret;
+
+ if (connection->last_poll == NULL) {
+ connection->last_poll = g_timer_new ();
+ } else if (!force &&
+ g_timer_elapsed (connection->last_poll, NULL) < POLL_INTERVAL) {
+ return CAM_RETURN_TRANSPORT_POLL;
+ }
+
+ GST_DEBUG ("polling connection %d", connection->id);
+ ret = cam_tl_connection_write_control_tpdu (connection, TAG_DATA_LAST);
+ if (CAM_FAILED (ret))
+ return ret;
+
+ g_timer_start (connection->last_poll);
+
+ return CAM_RETURN_OK;
+}
+
+/* read all the queued TPDUs */
+CamReturn
+cam_tl_read_all (CamTL * tl, gboolean poll)
+{
+ CamReturn ret = CAM_RETURN_OK;
+ CamTLConnection *connection;
+ GList *connections = NULL;
+ GList *walk;
+ gboolean done = FALSE;
+
+ while (!done) {
+ while (tl->expected_tpdus) {
+ /* read the next TPDU from the connection */
+ ret = cam_tl_read_tpdu_next (tl, &connection);
+ if (CAM_FAILED (ret)) {
+ GST_ERROR ("error reading TPDU from module: %d", ret);
+ goto out;
+ }
+
+ switch (tl->buffer[2]) {
+ case TAG_C_T_C_REPLY:
+ case TAG_D_T_C_REPLY:
+ connection->empty_data = 0;
+ ret = handle_control_tpdu (tl, connection);
+ break;
+ case TAG_DATA_MORE:
+ case TAG_DATA_LAST:
+ connection->empty_data = 0;
+ ret = handle_data_tpdu (tl, connection);
+ break;
+ case TAG_SB:
+ /* this is handled by tpdu_next */
+ break;
+ }
+
+ if (CAM_FAILED (ret))
+ goto out;
+ }
+
+ done = TRUE;
+
+ connections = NULL;
+ g_hash_table_foreach (tl->connections,
+ foreach_connection_get, &connections);
+
+ for (walk = connections; walk; walk = walk->next) {
+ CamTLConnection *connection = CAM_TL_CONNECTION (walk->data);
+
+ if (connection->has_data == TRUE && connection->empty_data < 10) {
+ ret = cam_tl_connection_write_control_tpdu (connection, TAG_RCV);
+ if (CAM_FAILED (ret)) {
+ g_list_free (connections);
+ goto out;
+ }
+ /* increment the empty_data counter. If we get data, this will be reset
+ * to 0 */
+ connection->empty_data++;
+ done = FALSE;
+ } else if (poll) {
+ ret = cam_tl_connection_poll (connection, FALSE);
+ if (ret == CAM_RETURN_TRANSPORT_POLL)
+ continue;
+
+ if (CAM_FAILED (ret)) {
+ g_list_free (connections);
+ goto out;
+ }
+
+ done = FALSE;
+ }
+ }
+
+ g_list_free (connections);
+ }
+
+out:
+ return ret;
+}
+
+CamReturn
+cam_tl_connection_write (CamTLConnection * connection,
+ guint8 * buffer, guint buffer_size, guint body_length)
+{
+ return cam_tl_connection_write_tpdu (connection,
+ TAG_DATA_LAST, buffer, buffer_size, 1 + body_length);
+}
diff --git a/sys/dvb/camtransport.h b/sys/dvb/camtransport.h
new file mode 100644
index 00000000..570b966c
--- /dev/null
+++ b/sys/dvb/camtransport.h
@@ -0,0 +1,115 @@
+/*
+ * camtransport.h - GStreamer CAM (EN50221) transport layer
+ * Copyright (C) 2007 Alessandro Decina
+ *
+ * Authors:
+ * Alessandro Decina <alessandro@nnva.org>
+ *
+ * This library is free software; you can redistribute it and/or
+ * modify it under the terms of the GNU Library General Public
+ * License as published by the Free Software Foundation; either
+ * version 2 of the License, or (at your option) any later version.
+ *
+ * This library 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
+ * Library General Public License for more details.
+ *
+ * You should have received a copy of the GNU Library General Public
+ * License along with this library; if not, write to the
+ * Free Software Foundation, Inc., 59 Temple Place - Suite 330,
+ * Boston, MA 02111-1307, USA.
+ */
+
+#ifndef CAM_TRANSPORT_H
+#define CAM_TRANSPORT_H
+
+#include <gst/gst.h>
+#include "cam.h"
+#include "camutils.h"
+
+#define HOST_BUFFER_SIZE 1024
+
+#define CAM_TL(obj) ((CamTL *) obj)
+#define CAM_TL_CONNECTION(obj) ((CamTLConnection *) obj)
+
+typedef struct _CamTL CamTL;
+typedef struct _CamTLPrivate CamTLPrivate;
+
+typedef struct _CamTLConnection CamTLConnection;
+typedef struct _CamTLConnectionPrivate CamTLConnectionPrivate;
+
+typedef enum
+{
+ CAM_TL_CONNECTION_STATE_CLOSED,
+ CAM_TL_CONNECTION_STATE_IN_CREATION,
+ CAM_TL_CONNECTION_STATE_OPEN,
+ CAM_TL_CONNECTION_STATE_IN_DELETION
+} CamTLConnectionState;
+
+struct _CamTL
+{
+ int fd;
+ guint connection_ids;
+
+ GHashTable *connections;
+
+ guint expected_tpdus;
+
+ /* buffer containing module data */
+ guint8 buffer [HOST_BUFFER_SIZE];
+ /* number of bytes written in the buffer */
+ guint buffer_size;
+ /* index pointing to the first byte of a TPDU's body */
+ guint8 *body;
+ /* length of the body part */
+ guint body_length;
+
+ /* callbacks */
+ void (*request_connection) (CamTL *tl, CamTLConnection *connection);
+ void (*connection_created) (CamTL *tl, CamTLConnection *connection);
+ void (*connection_deleted) (CamTL *tl, CamTLConnection *connection);
+ CamReturn (*connection_data) (CamTL *tl, CamTLConnection *connection,
+ guint8 *data, guint length);
+
+ /* used by the upper layer to extend this layer */
+ gpointer user_data;
+};
+
+struct _CamTLConnection
+{
+ CamTL *tl;
+
+ guint8 slot;
+ guint id;
+ CamTLConnectionState state;
+ /* TRUE if the last status byte was 0x80, FALSE otherwise */
+ gboolean has_data;
+ /* NCAS 1.0 sometimes reports that it has data even if it doesn't. After
+ * MAX_EMPTY_DATA times that we don't get any data we assume that there's
+ * actually no data.
+ */
+ guint empty_data;
+ /* timer restarted every time the connection is polled */
+ GTimer *last_poll;
+
+ gpointer user_data;
+};
+
+CamTL *cam_tl_new (int cam_device_fd);
+void cam_tl_destroy (CamTL *tl);
+
+CamReturn cam_tl_create_connection (CamTL *tl,
+ guint8 slot, CamTLConnection **connnection);
+CamReturn cam_tl_connection_delete (CamTLConnection *connection);
+
+void cam_tl_calc_buffer_size (CamTL *tl,
+ guint body_length, guint *buffer_size, guint *offset);
+
+CamReturn cam_tl_connection_write (CamTLConnection *connection,
+ guint8 *data, guint buffer_size, guint body_length);
+
+CamReturn cam_tl_connection_poll (CamTLConnection *connection, gboolean force);
+CamReturn cam_tl_read_all (CamTL *tl, gboolean poll);
+
+#endif /* CAM_TRANSPORT_H */
diff --git a/sys/dvb/camutils.c b/sys/dvb/camutils.c
new file mode 100644
index 00000000..7ca621b4
--- /dev/null
+++ b/sys/dvb/camutils.c
@@ -0,0 +1,340 @@
+/*
+ * camutils.c -
+ * Copyright (C) 2007 Alessandro Decina
+ *
+ * Authors:
+ * Alessandro Decina <alessandro@nnva.org>
+ *
+ * This library is free software; you can redistribute it and/or
+ * modify it under the terms of the GNU Library General Public
+ * License as published by the Free Software Foundation; either
+ * version 2 of the License, or (at your option) any later version.
+ *
+ * This library 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
+ * Library General Public License for more details.
+ *
+ * You should have received a copy of the GNU Library General Public
+ * License along with this library; if not, write to the
+ * Free Software Foundation, Inc., 59 Temple Place - Suite 330,
+ * Boston, MA 02111-1307, USA.
+ */
+
+#include <gst/gst.h>
+#include <string.h>
+
+#include "cam.h"
+#include "camutils.h"
+
+#define GST_CAT_DEFAULT cam_debug_cat
+
+/* From the spec:
+ * length_field() {
+ * size_indicator
+ * if (size_indicator == 0)
+ * length_value
+ * else if (size_indicator == 1) {
+ * length_field_size
+ * for (i=0; i<length_field_size; i++) {
+ * length_value_byte
+ * }
+ * }
+ * }
+*/
+
+guint8
+cam_calc_length_field_size (guint length)
+{
+ guint field_len;
+
+ if (length < G_MAXUINT8)
+ field_len = 1;
+ else if (length <= G_MAXUINT16)
+ field_len = 3;
+ else if (length <= (1 << 24) - 1)
+ field_len = 4;
+ else
+ field_len = 5;
+
+ return field_len;
+}
+
+/* write a length_field */
+guint8
+cam_write_length_field (guint8 * buff, guint length)
+{
+ guint8 field_len = cam_calc_length_field_size (length);
+
+ if (buff) {
+ switch (field_len) {
+ case 1:
+ buff[0] = length;
+ break;
+ case 2:
+ g_return_val_if_reached (0);
+ break;
+ case 3:
+ buff[0] = TPDU_HEADER_SIZE_INDICATOR | (field_len - 1);
+ buff[1] = length >> 8;
+ buff[2] = length & 0xFF;
+ break;
+ case 4:
+ buff[0] = TPDU_HEADER_SIZE_INDICATOR | (field_len - 1);
+ buff[1] = length >> 16;
+ buff[2] = (length >> 8) & 0xFF;
+ buff[3] = length & 0xFF;
+ break;
+ case 5:
+ buff[0] = TPDU_HEADER_SIZE_INDICATOR | (field_len - 1);
+ buff[1] = length >> 24;
+ buff[2] = (length >> 16) & 0xFF;
+ buff[3] = (length >> 8) & 0xFF;
+ buff[4] = length & 0xFF;
+ break;
+ default:
+ g_return_val_if_reached (0);
+ }
+ }
+
+ return field_len;
+}
+
+/* read a length_field */
+guint8
+cam_read_length_field (guint8 * buff, guint * length)
+{
+ guint i;
+ guint field_len;
+ guint8 len;
+
+ if (buff[0] <= G_MAXINT8) {
+ field_len = 1;
+ len = buff[0];
+ } else {
+ field_len = buff[0] & ~TPDU_HEADER_SIZE_INDICATOR;
+ if (field_len > 4) {
+ GST_ERROR ("length_field length exceeds 4 bytes: %d", field_len);
+ field_len = 0;
+ len = 0;
+ } else {
+ len = 0;
+ for (i = 0; i < field_len; ++i)
+ len = (len << 8) | *++buff;
+
+ /* count the size indicator byte */
+ field_len += 1;
+ }
+ }
+
+ if (length)
+ *length = len;
+
+ return field_len;
+}
+
+/*
+ * ca_pmt () {
+ * ca_pmt_tag 24 uimsbf
+ * length_field()
+ * ca_pmt_list_management 8 uimsbf
+ * program_number 16 uimsbf
+ * reserved 2 bslbf
+ * version_number 5 uimsbf
+ * current_next_indicator 1 bslbf
+ * reserved 4 bslbf
+ * program_info_length 12 uimsbf
+ * if (program_info_length != 0) {
+ * ca_pmt_cmd_id at program level 8 uimsbf
+ * for (i=0; i<n; i++) {
+ * CA_descriptor() programme level
+ * }
+ * }
+ * for (i=0; i<n; i++) {
+ * stream_type 8 uimsbf
+ * reserved 3 bslbf
+ * elementary_PID elementary stream PID 13 uimsbf
+ * reserved 4 bslbf
+ * ES_info_length 12 uimsbf
+ * if (ES_info_length != 0) {
+ * ca_pmt_cmd_id at ES level 8 uimsbf
+ * for (i=0; i<n; i++) {
+ * CA_descriptor() elementary stream level
+ * }
+ * }
+ * }
+ * }
+ */
+
+static guint
+get_ca_descriptors_length (GValueArray * descriptors)
+{
+ guint i;
+ guint len = 0;
+ GValue *value;
+ GString *desc;
+
+ if (descriptors != NULL) {
+ for (i = 0; i < descriptors->n_values; ++i) {
+ value = g_value_array_get_nth (descriptors, i);
+ desc = (GString *) g_value_get_boxed (value);
+
+ if (desc->str[0] == 0x09)
+ len += desc->len;
+ }
+ }
+
+ return len;
+}
+
+static guint8 *
+write_ca_descriptors (guint8 * body, GValueArray * descriptors)
+{
+ guint i;
+ GValue *value;
+ GString *desc;
+
+ if (descriptors != NULL) {
+ for (i = 0; i < descriptors->n_values; ++i) {
+ value = g_value_array_get_nth (descriptors, i);
+ desc = (GString *) g_value_get_boxed (value);
+
+ if (desc->str[0] == 0x09) {
+ memcpy (body, desc->str, desc->len);
+ body += desc->len;
+ }
+ }
+ }
+
+ return body;
+}
+
+guint8 *
+cam_build_ca_pmt (GObject * pmt, guint8 list_management, guint8 cmd_id,
+ guint * size)
+{
+ guint body_size = 0;
+ guint8 *buffer;
+ guint8 *body;
+ GList *lengths = NULL;
+ guint len = 0;
+ GValueArray *streams = NULL;
+ gint program_number;
+ guint version_number;
+ guint i;
+ GValue *value;
+ GObject *stream;
+ GValueArray *program_descriptors = NULL;
+ GValueArray *stream_descriptors = NULL;
+
+ g_object_get (pmt,
+ "program-number", &program_number,
+ "version-number", &version_number,
+ "stream-info", &streams, "descriptors", &program_descriptors, NULL);
+
+ /* get the length of program level CA_descriptor()s */
+ len = get_ca_descriptors_length (program_descriptors);
+ if (len > 0)
+ /* add one byte for the program level cmd_id */
+ len += 1;
+
+ lengths = g_list_append (lengths, GINT_TO_POINTER (len));
+ body_size += 6 + len;
+
+ /* get the length of stream level CA_descriptor()s */
+ if (streams != NULL) {
+ for (i = 0; i < streams->n_values; ++i) {
+ value = g_value_array_get_nth (streams, i);
+ stream = g_value_get_object (value);
+
+ g_object_get (stream, "descriptors", &stream_descriptors, NULL);
+
+ len = get_ca_descriptors_length (stream_descriptors);
+ if (len > 0)
+ /* one byte for the stream level cmd_id */
+ len += 1;
+
+ lengths = g_list_append (lengths, GINT_TO_POINTER (len));
+ body_size += 5 + len;
+
+ if (stream_descriptors) {
+ g_value_array_free (stream_descriptors);
+ stream_descriptors = NULL;
+ }
+ }
+ }
+
+ buffer = g_malloc0 (body_size);
+ body = buffer;
+
+ /*
+ guint length_field_len = cam_calc_length_field_size (body_size);
+ guint apdu_header_length = 3 + length_field_len;
+ guint8 *apdu = (buffer + buffer_size) - body_size - apdu_header_length;
+ apdu [0] = 0x9F;
+ apdu [1] = 0x80;
+ apdu [2] = 0x32;
+
+ g_print ("LEN %d %d", length_field_len, body_size);
+
+ cam_write_length_field (&apdu [3], body_size);
+ body += 4;
+ */
+
+ *body++ = list_management;
+
+ GST_WRITE_UINT16_BE (body, program_number);
+ body += 2;
+
+ *body++ = (version_number << 1) | 0x01;
+
+ len = GPOINTER_TO_INT (lengths->data);
+ lengths = g_list_delete_link (lengths, lengths);
+ GST_WRITE_UINT16_BE (body, len);
+ body += 2;
+
+ if (len != 0) {
+ *body++ = cmd_id;
+
+ body = write_ca_descriptors (body, program_descriptors);
+ }
+
+ if (program_descriptors)
+ g_value_array_free (program_descriptors);
+
+ for (i = 0; i < streams->n_values; ++i) {
+ guint stream_type;
+ guint stream_pid;
+
+ value = g_value_array_get_nth (streams, i);
+ stream = g_value_get_object (value);
+
+ g_object_get (stream,
+ "stream-type", &stream_type,
+ "pid", &stream_pid, "descriptors", &stream_descriptors, NULL);
+
+ *body++ = stream_type;
+ GST_WRITE_UINT16_BE (body, stream_pid);
+ body += 2;
+ len = GPOINTER_TO_INT (lengths->data);
+ lengths = g_list_delete_link (lengths, lengths);
+ GST_WRITE_UINT16_BE (body, len);
+ body += 2;
+
+ if (len != 0) {
+ *body++ = cmd_id;
+ body = write_ca_descriptors (body, stream_descriptors);
+ }
+
+ if (stream_descriptors) {
+ g_value_array_free (stream_descriptors);
+ stream_descriptors = NULL;
+ }
+ }
+
+ if (streams)
+ g_value_array_free (streams);
+
+ *size = body_size;
+ return buffer;
+}
diff --git a/sys/dvb/camutils.h b/sys/dvb/camutils.h
new file mode 100644
index 00000000..c63ba947
--- /dev/null
+++ b/sys/dvb/camutils.h
@@ -0,0 +1,59 @@
+/*
+ * camutils.h - GStreamer CAM (EN50221) support
+ * Copyright (C) 2007 Alessandro Decina
+ *
+ * Authors:
+ * Alessandro Decina <alessandro@nnva.org>
+ *
+ * This library is free software; you can redistribute it and/or
+ * modify it under the terms of the GNU Library General Public
+ * License as published by the Free Software Foundation; either
+ * version 2 of the License, or (at your option) any later version.
+ *
+ * This library 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
+ * Library General Public License for more details.
+ *
+ * You should have received a copy of the GNU Library General Public
+ * License along with this library; if not, write to the
+ * Free Software Foundation, Inc., 59 Temple Place - Suite 330,
+ * Boston, MA 02111-1307, USA.
+ */
+
+#ifndef CAM_UTILS_H
+#define CAM_UTILS_H
+
+#include <glib.h>
+#include <gst/gst.h>
+
+#define TPDU_HEADER_SIZE_INDICATOR 0x80
+
+#define CAM_FAILED(ret) (ret <= CAM_RETURN_ERROR)
+
+typedef enum
+{
+ /* generic */
+ CAM_RETURN_OK = 0,
+ CAM_RETURN_ERROR = -1,
+
+ /* transport specific */
+ CAM_RETURN_TRANSPORT_ERROR = -10,
+ CAM_RETURN_TRANSPORT_TOO_MANY_CONNECTIONS = -11,
+ CAM_RETURN_TRANSPORT_TIMEOUT = -12,
+ CAM_RETURN_TRANSPORT_POLL = -13,
+
+ /* session specific */
+ CAM_RETURN_SESSION_ERROR = -30,
+ CAM_RETURN_SESSION_TOO_MANY_SESSIONS = -31,
+
+ /* application specific */
+ CAM_RETURN_APPLICATION_ERROR = -40,
+} CamReturn;
+
+guint8 cam_calc_length_field_size (guint length);
+guint8 cam_write_length_field (guint8 *buff, guint length);
+guint8 cam_read_length_field (guint8 *buff, guint *length);
+guint8 *cam_build_ca_pmt (GObject *pmt, guint8 list_management, guint8 cmd_id, guint *size);
+
+#endif /* CAM_UTILS_H */
diff --git a/sys/dvb/dvbbasebin.c b/sys/dvb/dvbbasebin.c
new file mode 100644
index 00000000..ad043d63
--- /dev/null
+++ b/sys/dvb/dvbbasebin.c
@@ -0,0 +1,897 @@
+/*
+ * dvbbasebin.c -
+ * Copyright (C) 2007 Alessandro Decina
+ *
+ * Authors:
+ * Alessandro Decina <alessandro@nnva.org>
+ *
+ * This library is free software; you can redistribute it and/or
+ * modify it under the terms of the GNU Library General Public
+ * License as published by the Free Software Foundation; either
+ * version 2 of the License, or (at your option) any later version.
+ *
+ * This library 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
+ * Library General Public License for more details.
+ *
+ * You should have received a copy of the GNU Library General Public
+ * License along with this library; if not, write to the
+ * Free Software Foundation, Inc., 59 Temple Place - Suite 330,
+ * Boston, MA 02111-1307, USA.
+ */
+
+
+#ifdef HAVE_CONFIG_H
+#include "config.h"
+#endif
+#include <string.h>
+#include "dvbbasebin.h"
+
+GST_DEBUG_CATEGORY_STATIC (dvb_base_bin_debug);
+#define GST_CAT_DEFAULT dvb_base_bin_debug
+
+static GstElementDetails dvb_base_bin_details = GST_ELEMENT_DETAILS ("DVB bin",
+ "Source/Bin/Video",
+ "Access descramble and split DVB streams",
+ "Alessandro Decina <alessandro@nnva.org>");
+
+static GstStaticPadTemplate src_template =
+GST_STATIC_PAD_TEMPLATE ("src%d", GST_PAD_SRC,
+ GST_PAD_REQUEST,
+ GST_STATIC_CAPS ("video/mpegts, " "systemstream = (boolean) true ")
+ );
+
+static GstStaticPadTemplate program_template =
+GST_STATIC_PAD_TEMPLATE ("program_%d", GST_PAD_SRC,
+ GST_PAD_SOMETIMES,
+ GST_STATIC_CAPS ("video/mpegts, " "systemstream = (boolean) true ")
+ );
+
+enum
+{
+ /* FILL ME */
+ LAST_SIGNAL
+};
+
+enum
+{
+ ARG_0,
+ PROP_ADAPTER,
+ PROP_FRONTEND,
+ PROP_FREQUENCY,
+ PROP_POLARITY,
+ PROP_SYMBOL_RATE,
+ PROP_BANDWIDTH,
+ PROP_CODE_RATE_HP,
+ PROP_CODE_RATE_LP,
+ PROP_GUARD,
+ PROP_MODULATION,
+ PROP_TRANS_MODE,
+ PROP_HIERARCHY,
+ PROP_INVERSION,
+ PROP_PROGRAM_NUMBERS,
+ PROP_STATS_REPORTING_INTERVAL
+ /* FILL ME */
+};
+
+typedef struct
+{
+ guint16 pid;
+ guint usecount;
+} DvbBaseBinStream;
+
+typedef struct
+{
+ gint program_number;
+ guint16 pmt_pid;
+ guint16 pcr_pid;
+ GObject *pmt;
+ GObject *old_pmt;
+ gboolean selected;
+ gboolean pmt_active;
+ gboolean active;
+ GstPad *ghost;
+} DvbBaseBinProgram;
+
+static void dvb_base_bin_set_property (GObject * object, guint prop_id,
+ const GValue * value, GParamSpec * pspec);
+static void dvb_base_bin_get_property (GObject * object, guint prop_id,
+ GValue * value, GParamSpec * pspec);
+static void dvb_base_bin_dispose (GObject * object);
+static void dvb_base_bin_finalize (GObject * object);
+
+static gboolean dvb_base_bin_ts_pad_probe_cb (GstPad * pad,
+ GstBuffer * buf, gpointer user_data);
+static GstStateChangeReturn dvb_base_bin_change_state (GstElement * element,
+ GstStateChange transition);
+static void dvb_base_bin_pat_info_changed_cb (GstElement * mpegtsparse,
+ GParamSpec * pspec, DvbBaseBin * dvbbasebin);
+static void dvb_base_bin_pmt_info_cb (GstElement * mpegtsparse,
+ guint program_number, GObject * pmt, DvbBaseBin * dvbbasebin);
+static void dvb_base_bin_pad_added_cb (GstElement * mpegtsparse,
+ GstPad * pad, DvbBaseBin * dvbbasebin);
+static void dvb_base_bin_pad_removed_cb (GstElement * mpegtsparse,
+ GstPad * pad, DvbBaseBin * dvbbasebin);
+static GstPad *dvb_base_bin_request_new_pad (GstElement * element,
+ GstPadTemplate * templ, const gchar * name);
+static void dvb_base_bin_release_pad (GstElement * element, GstPad * pad);
+
+static DvbBaseBinStream *
+dvb_base_bin_add_stream (DvbBaseBin * dvbbasebin, guint16 pid)
+{
+ DvbBaseBinStream *stream;
+
+ stream = g_new0 (DvbBaseBinStream, 1);
+ stream->pid = pid;
+ stream->usecount = 0;
+
+ g_hash_table_insert (dvbbasebin->streams,
+ GINT_TO_POINTER ((gint) pid), stream);
+
+ return stream;
+}
+
+static DvbBaseBinStream *
+dvb_base_bin_get_stream (DvbBaseBin * dvbbasebin, guint16 pid)
+{
+ return (DvbBaseBinStream *) g_hash_table_lookup (dvbbasebin->streams,
+ GINT_TO_POINTER ((gint) pid));
+}
+
+static DvbBaseBinProgram *
+dvb_base_bin_add_program (DvbBaseBin * dvbbasebin, gint program_number)
+{
+ DvbBaseBinProgram *program;
+
+ program = g_new0 (DvbBaseBinProgram, 1);
+ program->program_number = program_number;
+ program->selected = FALSE;
+ program->active = FALSE;
+ program->pmt_pid = G_MAXUINT16;
+ program->pcr_pid = G_MAXUINT16;
+ program->pmt = NULL;
+ program->old_pmt = NULL;
+
+ g_hash_table_insert (dvbbasebin->programs,
+ GINT_TO_POINTER (program_number), program);
+
+ return program;
+}
+
+static DvbBaseBinProgram *
+dvb_base_bin_get_program (DvbBaseBin * dvbbasebin, gint program_number)
+{
+ return (DvbBaseBinProgram *) g_hash_table_lookup (dvbbasebin->programs,
+ GINT_TO_POINTER (program_number));
+}
+
+/*
+static guint signals [LAST_SIGNAL] = { 0 };
+*/
+
+GST_BOILERPLATE (DvbBaseBin, dvb_base_bin, GstBin, GST_TYPE_BIN);
+
+static void
+dvb_base_bin_base_init (gpointer klass)
+{
+ GstElementClass *element_class = GST_ELEMENT_CLASS (klass);
+
+ element_class->request_new_pad = dvb_base_bin_request_new_pad;
+ element_class->release_pad = dvb_base_bin_release_pad;
+
+ gst_element_class_add_pad_template (element_class,
+ gst_static_pad_template_get (&src_template));
+ gst_element_class_add_pad_template (element_class,
+ gst_static_pad_template_get (&program_template));
+
+ gst_element_class_set_details (element_class, &dvb_base_bin_details);
+}
+
+static void
+dvb_base_bin_class_init (DvbBaseBinClass * klass)
+{
+ GObjectClass *gobject_class;
+ GstElementClass *element_class;
+ GstElementFactory *dvbsrc_factory;
+ GObjectClass *dvbsrc_class;
+ typedef struct
+ {
+ guint prop_id;
+ const gchar *prop_name;
+ } ProxyedProperty;
+ ProxyedProperty *walk;
+ ProxyedProperty proxyed_properties[] = {
+ {PROP_ADAPTER, "adapter"},
+ {PROP_FRONTEND, "frontend"},
+ {PROP_FREQUENCY, "frequency"},
+ {PROP_POLARITY, "polarity"},
+ {PROP_SYMBOL_RATE, "symbol-rate"},
+ {PROP_BANDWIDTH, "bandwidth"},
+ {PROP_CODE_RATE_HP, "code-rate-hp"},
+ {PROP_CODE_RATE_LP, "code-rate-lp"},
+ {PROP_GUARD, "guard"},
+ {PROP_MODULATION, "modulation"},
+ {PROP_TRANS_MODE, "trans-mode"},
+ {PROP_HIERARCHY, "hierarchy"},
+ {PROP_INVERSION, "inversion"},
+ {PROP_STATS_REPORTING_INTERVAL, "stats-reporting-interval"},
+ {0, NULL}
+ };
+
+ element_class = GST_ELEMENT_CLASS (klass);
+ element_class->change_state = dvb_base_bin_change_state;
+
+ gobject_class = G_OBJECT_CLASS (klass);
+ gobject_class->set_property = dvb_base_bin_set_property;
+ gobject_class->get_property = dvb_base_bin_get_property;
+ gobject_class->dispose = dvb_base_bin_dispose;
+ gobject_class->finalize = dvb_base_bin_finalize;
+
+ /* install dvbsrc properties */
+ dvbsrc_factory = gst_element_factory_find ("dvbsrc");
+ dvbsrc_class = g_type_class_ref (dvbsrc_factory->type);
+ walk = proxyed_properties;
+ while (walk->prop_name != NULL) {
+ GParamSpec *pspec;
+ GParamSpec *our_pspec;
+
+ pspec = g_object_class_find_property (dvbsrc_class, walk->prop_name);
+ if (pspec != NULL) {
+ GType param_type = G_PARAM_SPEC_TYPE (pspec);
+
+ if (param_type == G_TYPE_PARAM_INT) {
+ GParamSpecInt *src_pspec = G_PARAM_SPEC_INT (pspec);
+
+ our_pspec = g_param_spec_int (g_param_spec_get_name (pspec),
+ g_param_spec_get_nick (pspec), g_param_spec_get_blurb (pspec),
+ src_pspec->minimum, src_pspec->maximum, src_pspec->default_value,
+ pspec->flags);
+ } else if (param_type == G_TYPE_PARAM_UINT) {
+ GParamSpecUInt *src_pspec = G_PARAM_SPEC_UINT (pspec);
+
+ our_pspec = g_param_spec_uint (g_param_spec_get_name (pspec),
+ g_param_spec_get_nick (pspec), g_param_spec_get_blurb (pspec),
+ src_pspec->minimum, src_pspec->maximum, src_pspec->default_value,
+ pspec->flags);
+ } else if (param_type == G_TYPE_PARAM_STRING) {
+ GParamSpecString *src_pspec = G_PARAM_SPEC_STRING (pspec);
+
+ our_pspec = g_param_spec_string (g_param_spec_get_name (pspec),
+ g_param_spec_get_nick (pspec), g_param_spec_get_blurb (pspec),
+ src_pspec->default_value, pspec->flags);
+ } else if (param_type == G_TYPE_PARAM_ENUM) {
+ GParamSpecEnum *src_pspec = G_PARAM_SPEC_ENUM (pspec);
+
+ our_pspec = g_param_spec_enum (g_param_spec_get_name (pspec),
+ g_param_spec_get_nick (pspec), g_param_spec_get_blurb (pspec),
+ pspec->value_type, src_pspec->default_value, pspec->flags);
+ } else {
+ g_assert_not_reached ();
+ }
+
+ g_object_class_install_property (gobject_class, walk->prop_id, our_pspec);
+ } else {
+ g_warning ("dvbsrc has no property named %s", walk->prop_name);
+ }
+ ++walk;
+ }
+ g_type_class_unref (dvbsrc_class);
+
+ g_object_class_install_property (gobject_class, PROP_PROGRAM_NUMBERS,
+ g_param_spec_string ("program-numbers",
+ "Program Numbers",
+ "Colon separated list of programs", "", G_PARAM_READWRITE));
+}
+
+static void
+dvb_base_bin_reset (DvbBaseBin * dvbbasebin)
+{
+ if (dvbbasebin->hwcam) {
+ cam_device_close (dvbbasebin->hwcam);
+ cam_device_free (dvbbasebin->hwcam);
+ dvbbasebin->hwcam = NULL;
+ }
+
+ if (dvbbasebin->ts_pad) {
+ gst_element_release_request_pad (GST_ELEMENT (dvbbasebin->mpegtsparse),
+ dvbbasebin->ts_pad);
+ dvbbasebin->ts_pad = NULL;
+ }
+}
+
+static void
+dvb_base_bin_init (DvbBaseBin * dvbbasebin, DvbBaseBinClass * klass)
+{
+ dvbbasebin->dvbsrc = gst_element_factory_make ("dvbsrc", NULL);
+ dvbbasebin->buffer_queue = gst_element_factory_make ("queue", NULL);
+ dvbbasebin->mpegtsparse = gst_element_factory_make ("mpegtsparse", NULL);
+ g_object_connect (dvbbasebin->mpegtsparse,
+ "signal::notify::pat-info", dvb_base_bin_pat_info_changed_cb, dvbbasebin,
+ "signal::pmt-info", dvb_base_bin_pmt_info_cb, dvbbasebin,
+ "signal::pad-added", dvb_base_bin_pad_added_cb, dvbbasebin,
+ "signal::pad-removed", dvb_base_bin_pad_removed_cb, dvbbasebin, NULL);
+
+ gst_bin_add_many (GST_BIN (dvbbasebin), dvbbasebin->dvbsrc,
+ dvbbasebin->buffer_queue, dvbbasebin->mpegtsparse, NULL);
+
+ gst_element_link_many (dvbbasebin->dvbsrc,
+ dvbbasebin->buffer_queue, dvbbasebin->mpegtsparse, NULL);
+
+ dvbbasebin->programs = g_hash_table_new_full (g_direct_hash, g_direct_equal,
+ NULL, g_free);
+ dvbbasebin->streams = g_hash_table_new_full (g_direct_hash, g_direct_equal,
+ NULL, g_free);
+
+ dvbbasebin->pmtlist = NULL;
+ dvbbasebin->pmtlist_changed = FALSE;
+
+ dvbbasebin->disposed = FALSE;
+ dvb_base_bin_reset (dvbbasebin);
+}
+
+static void
+dvb_base_bin_dispose (GObject * object)
+{
+ DvbBaseBin *dvbbasebin = GST_DVB_BASE_BIN (object);
+
+ if (!dvbbasebin->disposed) {
+ /* remove mpegtsparse BEFORE dvbsrc, since the mpegtsparse::pad-removed
+ * signal handler uses dvbsrc */
+ dvb_base_bin_reset (dvbbasebin);
+ gst_bin_remove (GST_BIN (dvbbasebin), dvbbasebin->mpegtsparse);
+ gst_bin_remove (GST_BIN (dvbbasebin), dvbbasebin->dvbsrc);
+ gst_bin_remove (GST_BIN (dvbbasebin), dvbbasebin->buffer_queue);
+ dvbbasebin->disposed = TRUE;
+ }
+
+ if (G_OBJECT_CLASS (parent_class)->dispose)
+ G_OBJECT_CLASS (parent_class)->dispose (object);
+}
+
+static void
+dvb_base_bin_finalize (GObject * object)
+{
+ DvbBaseBin *dvbbasebin = GST_DVB_BASE_BIN (object);
+
+ g_hash_table_destroy (dvbbasebin->streams);
+ g_hash_table_destroy (dvbbasebin->programs);
+
+ if (G_OBJECT_CLASS (parent_class)->finalize)
+ G_OBJECT_CLASS (parent_class)->finalize (object);
+}
+
+static void
+dvb_base_bin_set_property (GObject * object, guint prop_id,
+ const GValue * value, GParamSpec * pspec)
+{
+ DvbBaseBin *dvbbasebin = GST_DVB_BASE_BIN (object);
+
+ switch (prop_id) {
+ case PROP_ADAPTER:
+ case PROP_FRONTEND:
+ case PROP_FREQUENCY:
+ case PROP_POLARITY:
+ case PROP_SYMBOL_RATE:
+ case PROP_BANDWIDTH:
+ case PROP_CODE_RATE_HP:
+ case PROP_CODE_RATE_LP:
+ case PROP_GUARD:
+ case PROP_MODULATION:
+ case PROP_TRANS_MODE:
+ case PROP_HIERARCHY:
+ case PROP_INVERSION:
+ case PROP_STATS_REPORTING_INTERVAL:
+ /* FIXME: check if we can tune (state < PLAYING || program-numbers == "") */
+ g_object_set_property (G_OBJECT (dvbbasebin->dvbsrc), pspec->name, value);
+ break;
+ case PROP_PROGRAM_NUMBERS:
+ g_object_set_property (G_OBJECT (dvbbasebin->mpegtsparse), pspec->name,
+ value);
+ break;
+ default:
+ G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
+ }
+}
+
+static void
+dvb_base_bin_get_property (GObject * object, guint prop_id,
+ GValue * value, GParamSpec * pspec)
+{
+ DvbBaseBin *dvbbasebin = GST_DVB_BASE_BIN (object);
+
+ switch (prop_id) {
+ case PROP_ADAPTER:
+ case PROP_FRONTEND:
+ case PROP_FREQUENCY:
+ case PROP_POLARITY:
+ case PROP_SYMBOL_RATE:
+ case PROP_BANDWIDTH:
+ case PROP_CODE_RATE_HP:
+ case PROP_CODE_RATE_LP:
+ case PROP_GUARD:
+ case PROP_MODULATION:
+ case PROP_TRANS_MODE:
+ case PROP_HIERARCHY:
+ case PROP_INVERSION:
+ case PROP_STATS_REPORTING_INTERVAL:
+ g_object_get_property (G_OBJECT (dvbbasebin->dvbsrc), pspec->name, value);
+ break;
+ case PROP_PROGRAM_NUMBERS:
+ g_object_get_property (G_OBJECT (dvbbasebin->mpegtsparse), pspec->name,
+ value);
+ break;
+ default:
+ G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
+ }
+}
+
+static GstPad *
+dvb_base_bin_request_new_pad (GstElement * element,
+ GstPadTemplate * templ, const gchar * name)
+{
+ GstPad *pad;
+ GstPad *ghost;
+ gchar *pad_name;
+
+ if (name == NULL)
+ name = GST_PAD_TEMPLATE_NAME_TEMPLATE (templ);
+
+ pad =
+ gst_element_get_request_pad (GST_DVB_BASE_BIN (element)->mpegtsparse,
+ name);
+ if (pad == NULL)
+ return NULL;
+
+ pad_name = gst_pad_get_name (pad);
+ ghost = gst_ghost_pad_new (pad_name, pad);
+ g_free (pad_name);
+ gst_element_add_pad (element, ghost);
+
+ return ghost;
+}
+
+static void
+dvb_base_bin_release_pad (GstElement * element, GstPad * pad)
+{
+ GstGhostPad *ghost;
+ GstPad *target;
+
+ g_return_if_fail (GST_IS_DVB_BASE_BIN (element));
+
+ ghost = GST_GHOST_PAD (pad);
+ target = gst_ghost_pad_get_target (ghost);
+ gst_element_release_request_pad (GST_ELEMENT (GST_DVB_BASE_BIN (element)->
+ mpegtsparse), target);
+ gst_object_unref (target);
+
+ gst_element_remove_pad (element, pad);
+}
+
+static void
+dvb_base_bin_reset_pmtlist (DvbBaseBin * dvbbasebin)
+{
+ CamConditionalAccessPmtFlag flag;
+ GList *walk;
+ GObject *pmt;
+
+ walk = dvbbasebin->pmtlist;
+ while (walk) {
+ if (walk->prev == NULL) {
+ if (walk->next == NULL)
+ flag = CAM_CONDITIONAL_ACCESS_PMT_FLAG_ONLY;
+ else
+ flag = CAM_CONDITIONAL_ACCESS_PMT_FLAG_FIRST;
+ } else {
+ if (walk->next == NULL)
+ flag = CAM_CONDITIONAL_ACCESS_PMT_FLAG_LAST;
+ else
+ flag = CAM_CONDITIONAL_ACCESS_PMT_FLAG_MORE;
+ }
+
+ pmt = G_OBJECT (walk->data);
+ cam_device_set_pmt (dvbbasebin->hwcam, pmt, flag);
+
+ walk = walk->next;
+ }
+
+ dvbbasebin->pmtlist_changed = FALSE;
+}
+
+static gboolean
+dvb_base_bin_ts_pad_probe_cb (GstPad * pad, GstBuffer * buf, gpointer user_data)
+{
+ DvbBaseBin *dvbbasebin = GST_DVB_BASE_BIN (user_data);
+
+ if (dvbbasebin->hwcam) {
+ cam_device_poll (dvbbasebin->hwcam);
+
+ if (dvbbasebin->pmtlist_changed) {
+ if (cam_device_ready (dvbbasebin->hwcam)) {
+ GST_DEBUG_OBJECT (dvbbasebin, "pmt list changed");
+ dvb_base_bin_reset_pmtlist (dvbbasebin);
+ } else {
+ GST_DEBUG_OBJECT (dvbbasebin, "pmt list changed but CAM not ready");
+ }
+ }
+ }
+
+ return TRUE;
+}
+
+static void
+dvb_base_bin_init_cam (DvbBaseBin * dvbbasebin)
+{
+ gint adapter;
+ gchar *ca_file;
+
+ g_object_get (dvbbasebin->dvbsrc, "adapter", &adapter, NULL);
+ /* TODO: handle multiple cams */
+ ca_file = g_strdup_printf ("/dev/dvb/adapter%d/ca0", adapter);
+ if (g_file_test (ca_file, G_FILE_TEST_EXISTS)) {
+ dvbbasebin->hwcam = cam_device_new ();
+ if (cam_device_open (dvbbasebin->hwcam, ca_file)) {
+ /* HACK: poll the cam in a buffer probe */
+ dvbbasebin->ts_pad =
+ gst_element_get_request_pad (dvbbasebin->mpegtsparse, "src%d");
+ gst_pad_add_buffer_probe (dvbbasebin->ts_pad,
+ G_CALLBACK (dvb_base_bin_ts_pad_probe_cb), dvbbasebin);
+ } else {
+ GST_ERROR_OBJECT (dvbbasebin, "could not open %s", ca_file);
+ cam_device_free (dvbbasebin->hwcam);
+ dvbbasebin->hwcam = NULL;
+ }
+ }
+
+ g_free (ca_file);
+}
+
+static GstStateChangeReturn
+dvb_base_bin_change_state (GstElement * element, GstStateChange transition)
+{
+ DvbBaseBin *dvbbasebin;
+ GstStateChangeReturn ret;
+
+ dvbbasebin = GST_DVB_BASE_BIN (element);
+ ret = GST_ELEMENT_CLASS (parent_class)->change_state (element, transition);
+
+ switch (transition) {
+ case GST_STATE_CHANGE_READY_TO_PAUSED:
+ dvb_base_bin_init_cam (dvbbasebin);
+ break;
+ case GST_STATE_CHANGE_PAUSED_TO_READY:
+ dvb_base_bin_reset (dvbbasebin);
+ break;
+ default:
+ break;
+ }
+
+ return ret;
+}
+
+static void
+foreach_stream_build_filter (gpointer key, gpointer value, gpointer user_data)
+{
+ DvbBaseBin *dvbbasebin = GST_DVB_BASE_BIN (user_data);
+ DvbBaseBinStream *stream = (DvbBaseBinStream *) value;
+ gchar *tmp, *pid;
+
+ g_assert (stream->usecount >= 0);
+
+ GST_DEBUG ("stream %d usecount %d", stream->pid, stream->usecount);
+
+ if (stream->usecount > 0) {
+ /* TODO: use g_strjoinv FTW */
+ tmp = dvbbasebin->filter;
+ pid = g_strdup_printf ("%d", stream->pid);
+ dvbbasebin->filter = g_strjoin (":", pid, dvbbasebin->filter, NULL);
+
+ g_free (pid);
+ g_free (tmp);
+ }
+}
+
+static void
+dvb_base_bin_rebuild_filter (DvbBaseBin * dvbbasebin)
+{
+ g_hash_table_foreach (dvbbasebin->streams,
+ foreach_stream_build_filter, dvbbasebin);
+
+ if (dvbbasebin->filter == NULL)
+ /* fix dvbsrc to handle NULL filter */
+ dvbbasebin->filter = g_strdup ("");
+
+ GST_INFO_OBJECT (dvbbasebin, "rebuilt filter %s", dvbbasebin->filter);
+
+ /* FIXME: disable this until we find a way to filter out unwanted pids
+ g_object_set (dvbbasebin->dvbsrc, "pids", dvbbasebin->filter, NULL);
+ */
+ g_free (dvbbasebin->filter);
+ dvbbasebin->filter = NULL;
+}
+
+static void
+dvb_base_bin_remove_pmt_streams (DvbBaseBin * dvbbasebin, GObject * pmt)
+{
+ GValueArray *streams;
+ gint program_number;
+ gint i;
+ GValue *value;
+ GObject *streamobj;
+ DvbBaseBinStream *stream;
+ guint pid;
+ guint stream_type;
+
+ g_object_get (pmt, "program-number", &program_number,
+ "stream-info", &streams, NULL);
+
+ for (i = 0; i < streams->n_values; ++i) {
+ value = g_value_array_get_nth (streams, i);
+ streamobj = g_value_get_object (value);
+
+ g_object_get (streamobj, "pid", &pid, "stream-type", &stream_type, NULL);
+
+ stream = dvb_base_bin_get_stream (dvbbasebin, (guint16) pid);
+ if (stream == NULL) {
+ GST_WARNING_OBJECT (dvbbasebin, "removing unknown stream %d ??", pid);
+ continue;
+ }
+
+ --stream->usecount;
+ }
+}
+
+static void
+dvb_base_bin_add_pmt_streams (DvbBaseBin * dvbbasebin, GObject * pmt)
+{
+ DvbBaseBinStream *stream;
+ GValueArray *streams;
+ gint program_number;
+ gint i;
+ GValue *value;
+ GObject *streamobj;
+ guint pid;
+ guint stream_type;
+
+ g_object_get (pmt, "program-number", &program_number,
+ "stream-info", &streams, NULL);
+
+ for (i = 0; i < streams->n_values; ++i) {
+ value = g_value_array_get_nth (streams, i);
+ streamobj = g_value_get_object (value);
+
+ g_object_get (streamobj, "pid", &pid, "stream-type", &stream_type, NULL);
+ GST_DEBUG ("filtering stream %d stream_type %d", pid, stream_type);
+
+ stream = dvb_base_bin_get_stream (dvbbasebin, (guint16) pid);
+ if (stream == NULL)
+ stream = dvb_base_bin_add_stream (dvbbasebin, (guint16) pid);
+
+ ++stream->usecount;
+ }
+}
+
+static void
+dvb_base_bin_activate_program (DvbBaseBin * dvbbasebin,
+ DvbBaseBinProgram * program)
+{
+ DvbBaseBinStream *stream;
+ guint pid;
+
+ if (program->old_pmt) {
+ dvb_base_bin_remove_pmt_streams (dvbbasebin, program->old_pmt);
+ dvbbasebin->pmtlist = g_list_remove (dvbbasebin->pmtlist, program->old_pmt);
+ }
+
+ /* activate the PMT and PCR streams. If the PCR stream is in the PMT its
+ * usecount will be incremented by 2 here and decremented by 2 when the
+ * program is deactivated */
+ if (!program->pmt_active) {
+ stream = dvb_base_bin_get_stream (dvbbasebin, program->pmt_pid);
+ if (stream == NULL)
+ stream = dvb_base_bin_add_stream (dvbbasebin, program->pmt_pid);
+ stream->usecount += 1;
+ program->pmt_active = TRUE;
+ }
+
+ if (program->pmt) {
+ guint16 old_pcr_pid;
+
+ old_pcr_pid = program->pcr_pid;
+ g_object_get (program->pmt, "pcr-pid", &pid, NULL);
+ program->pcr_pid = pid;
+ if (old_pcr_pid != G_MAXUINT16 && old_pcr_pid != program->pcr_pid)
+ dvb_base_bin_get_stream (dvbbasebin, old_pcr_pid)->usecount--;
+
+ stream = dvb_base_bin_get_stream (dvbbasebin, program->pcr_pid);
+ if (stream == NULL)
+ stream = dvb_base_bin_add_stream (dvbbasebin, program->pcr_pid);
+ stream->usecount += 1;
+
+ dvb_base_bin_add_pmt_streams (dvbbasebin, program->pmt);
+ dvbbasebin->pmtlist = g_list_append (dvbbasebin->pmtlist, program->pmt);
+ dvbbasebin->pmtlist_changed = TRUE;
+ program->active = TRUE;
+ }
+
+ dvb_base_bin_rebuild_filter (dvbbasebin);
+}
+
+static void
+dvb_base_bin_deactivate_program (DvbBaseBin * dvbbasebin,
+ DvbBaseBinProgram * program)
+{
+ DvbBaseBinStream *stream;
+
+ stream = dvb_base_bin_get_stream (dvbbasebin, program->pmt_pid);
+ if (stream != NULL)
+ stream->usecount -= 1;
+
+ stream = dvb_base_bin_get_stream (dvbbasebin, program->pcr_pid);
+ if (stream != NULL)
+ stream->usecount -= 1;
+
+ if (program->pmt) {
+ dvb_base_bin_remove_pmt_streams (dvbbasebin, program->pmt);
+ dvbbasebin->pmtlist = g_list_remove (dvbbasebin->pmtlist, program->pmt);
+ dvbbasebin->pmtlist_changed = TRUE;
+ }
+
+ dvb_base_bin_rebuild_filter (dvbbasebin);
+ program->pmt_active = FALSE;
+ program->active = FALSE;
+}
+
+static void
+dvb_base_bin_pat_info_changed_cb (GstElement * mpegtsparse,
+ GParamSpec * pspec, DvbBaseBin * dvbbasebin)
+{
+ DvbBaseBinProgram *program;
+ DvbBaseBinStream *stream;
+ GValueArray *pat_info;
+ GValue *value;
+ GObject *program_info;
+ gint program_number;
+ guint pid;
+ guint16 old_pmt_pid;
+ gint i;
+ gboolean rebuild_filter = FALSE;
+
+ g_object_get (mpegtsparse, "pat-info", &pat_info, NULL);
+
+ for (i = 0; i < pat_info->n_values; ++i) {
+ value = g_value_array_get_nth (pat_info, i);
+ program_info = g_value_get_object (value);
+
+ g_object_get (program_info, "program-number", &program_number,
+ "pid", &pid, NULL);
+
+ program = dvb_base_bin_get_program (dvbbasebin, program_number);
+ if (program == NULL)
+ program = dvb_base_bin_add_program (dvbbasebin, program_number);
+
+ old_pmt_pid = program->pmt_pid;
+ program->pmt_pid = pid;
+
+ if (program->selected) {
+ /* PAT update */
+ if (old_pmt_pid != G_MAXUINT16 && old_pmt_pid != program->pmt_pid)
+ dvb_base_bin_get_stream (dvbbasebin, old_pmt_pid)->usecount -= 1;
+
+ stream = dvb_base_bin_get_stream (dvbbasebin, program->pmt_pid);
+ if (stream == NULL)
+ stream = dvb_base_bin_add_stream (dvbbasebin, program->pmt_pid);
+
+ stream->usecount += 1;
+
+ rebuild_filter = TRUE;
+ }
+ }
+
+ g_value_array_free (pat_info);
+
+ if (rebuild_filter)
+ dvb_base_bin_rebuild_filter (dvbbasebin);
+}
+
+static void
+dvb_base_bin_pmt_info_cb (GstElement * mpegtsparse,
+ guint program_number, GObject * pmt, DvbBaseBin * dvbbasebin)
+{
+ DvbBaseBinProgram *program;
+
+ program = dvb_base_bin_get_program (dvbbasebin, program_number);
+ if (program == NULL) {
+ GST_WARNING ("got PMT for program %d but program not in PAT",
+ program_number);
+ program = dvb_base_bin_add_program (dvbbasebin, program_number);
+ }
+
+ program->old_pmt = program->pmt;
+ program->pmt = g_object_ref (pmt);
+
+ /* activate the program if it's selected and either it's not active or its pmt
+ * changed */
+ if (program->selected && (!program->active || program->old_pmt != NULL))
+ dvb_base_bin_activate_program (dvbbasebin, program);
+
+ if (program->old_pmt) {
+ g_object_unref (program->old_pmt);
+ program->old_pmt = NULL;
+ }
+}
+
+static gint
+get_pad_program_number (GstPad * pad)
+{
+ gchar *progstr;
+ gchar *name;
+
+ name = gst_pad_get_name (pad);
+
+ if (strncmp (name, "program_", 8) != 0) {
+ g_free (name);
+ return -1;
+ }
+
+ progstr = strstr (name, "_");
+ g_free (name);
+ if (progstr == NULL)
+ return -1;
+
+ return strtol (++progstr, NULL, 10);
+}
+
+static void
+dvb_base_bin_pad_added_cb (GstElement * mpegtsparse,
+ GstPad * pad, DvbBaseBin * dvbbasebin)
+{
+ DvbBaseBinProgram *program;
+ gint program_number;
+
+ program_number = get_pad_program_number (pad);
+ if (program_number == -1)
+ return;
+
+ program = dvb_base_bin_get_program (dvbbasebin, program_number);
+ if (program == NULL)
+ program = dvb_base_bin_add_program (dvbbasebin, program_number);
+ program->selected = TRUE;
+ program->ghost = gst_ghost_pad_new (gst_pad_get_name (pad), pad);
+ gst_pad_set_active (program->ghost, TRUE);
+ gst_element_add_pad (GST_ELEMENT (dvbbasebin), program->ghost);
+
+ /* if the program has a pmt, activate it now, otherwise it will get activated
+ * when there's a PMT */
+ if (!program->active && program->pmt_pid != G_MAXUINT16)
+ dvb_base_bin_activate_program (dvbbasebin, program);
+}
+
+static void
+dvb_base_bin_pad_removed_cb (GstElement * mpegtsparse,
+ GstPad * pad, DvbBaseBin * dvbbasebin)
+{
+ DvbBaseBinProgram *program;
+ gint program_number;
+
+ program_number = get_pad_program_number (pad);
+ if (program_number == -1)
+ return;
+
+ program = dvb_base_bin_get_program (dvbbasebin, program_number);
+ program->selected = FALSE;
+ dvb_base_bin_deactivate_program (dvbbasebin, program);
+ gst_element_remove_pad (GST_ELEMENT (dvbbasebin), program->ghost);
+ program->ghost = NULL;
+}
+
+gboolean
+gst_dvb_base_bin_plugin_init (GstPlugin * plugin)
+{
+ GST_DEBUG_CATEGORY_INIT (dvb_base_bin_debug, "dvbbasebin", 0, "DVB bin");
+
+ cam_init ();
+
+ return gst_element_register (plugin, "dvbbasebin",
+ GST_RANK_NONE, GST_TYPE_DVB_BASE_BIN);
+}
diff --git a/sys/dvb/dvbbasebin.h b/sys/dvb/dvbbasebin.h
new file mode 100644
index 00000000..fbf0b189
--- /dev/null
+++ b/sys/dvb/dvbbasebin.h
@@ -0,0 +1,75 @@
+/*
+ * dvbbasebin.h -
+ * Copyright (C) 2007 Alessandro Decina
+ *
+ * Authors:
+ * Alessandro Decina <alessandro@nnva.org>
+ *
+ * This library is free software; you can redistribute it and/or
+ * modify it under the terms of the GNU Library General Public
+ * License as published by the Free Software Foundation; either
+ * version 2 of the License, or (at your option) any later version.
+ *
+ * This library 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
+ * Library General Public License for more details.
+ *
+ * You should have received a copy of the GNU Library General Public
+ * License along with this library; if not, write to the
+ * Free Software Foundation, Inc., 59 Temple Place - Suite 330,
+ * Boston, MA 02111-1307, USA.
+ */
+
+#ifndef GST_DVB_BASE_BIN_H
+#define GST_DVB_BASE_BIN_H
+
+#include <gst/gst.h>
+#include <glib.h>
+#include "camdevice.h"
+
+G_BEGIN_DECLS
+
+#define GST_TYPE_DVB_BASE_BIN \
+ (dvb_base_bin_get_type())
+#define GST_DVB_BASE_BIN(obj) \
+ (G_TYPE_CHECK_INSTANCE_CAST((obj),GST_TYPE_DVB_BASE_BIN,DvbBaseBin))
+#define GST_DVB_BASE_BIN_CLASS(klass) \
+ (G_TYPE_CHECK_CLASS_CAST((klass),GST_TYPE_DVB_BASE_BIN,DvbBaseBinClass))
+#define GST_IS_DVB_BASE_BIN(obj) \
+ (G_TYPE_CHECK_INSTANCE_TYPE((obj),GST_TYPE_DVB_BASE_BIN))
+#define GST_IS_DVB_BASE_BIN_CLASS(klass) \
+ (G_TYPE_CHECK_CLASS_TYPE((klass),GST_TYPE_DVB_BASE_BIN))
+
+typedef struct _DvbBaseBin DvbBaseBin;
+typedef struct _DvbBaseBinClass DvbBaseBinClass;
+
+struct _DvbBaseBin {
+ GstBin bin;
+
+ GstPad *ts_pad;
+
+ GstElement *dvbsrc;
+ GstElement *buffer_queue;
+ GstElement *mpegtsparse;
+ CamDevice *hwcam;
+ GList *pmtlist;
+ gboolean pmtlist_changed;
+ gchar *filter;
+ GHashTable *streams;
+ GHashTable *programs;
+ gboolean disposed;
+};
+
+struct _DvbBaseBinClass {
+ GstBinClass parent_class;
+
+ /* signals */
+};
+
+GType gst_dvb_base_bin_get_type(void);
+gboolean gst_dvb_base_bin_plugin_init (GstPlugin *plugin);
+
+G_END_DECLS
+
+#endif /* GST_DVB_BASE_BIN_H */
diff --git a/sys/dvb/gstdvb.c b/sys/dvb/gstdvb.c
new file mode 100644
index 00000000..1087173e
--- /dev/null
+++ b/sys/dvb/gstdvb.c
@@ -0,0 +1,46 @@
+/*
+ * gstdvb.c -
+ * Copyright (C) 2007 Alessandro Decina
+ *
+ * Authors:
+ * Alessandro Decina <alessandro@nnva.org>
+ *
+ * This library is free software; you can redistribute it and/or
+ * modify it under the terms of the GNU Library General Public
+ * License as published by the Free Software Foundation; either
+ * version 2 of the License, or (at your option) any later version.
+ *
+ * This library 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
+ * Library General Public License for more details.
+ *
+ * You should have received a copy of the GNU Library General Public
+ * License along with this library; if not, write to the
+ * Free Software Foundation, Inc., 59 Temple Place - Suite 330,
+ * Boston, MA 02111-1307, USA.
+ */
+#ifdef HAVE_CONFIG_H
+#include "config.h"
+#endif
+
+#include "gstdvbsrc.h"
+#include "dvbbasebin.h"
+
+static gboolean
+plugin_init (GstPlugin * plugin)
+{
+ if (!gst_dvbsrc_plugin_init (plugin))
+ return FALSE;
+
+ if (!gst_dvb_base_bin_plugin_init (plugin))
+ return FALSE;
+
+ return TRUE;
+}
+
+GST_PLUGIN_DEFINE (GST_VERSION_MAJOR,
+ GST_VERSION_MINOR,
+ "dvb",
+ "DVB elements",
+ plugin_init, VERSION, GST_LICENSE, GST_PACKAGE_NAME, GST_PACKAGE_ORIGIN)
diff --git a/sys/dvb/gstdvbsrc.c b/sys/dvb/gstdvbsrc.c
index f72ed704..84594527 100644
--- a/sys/dvb/gstdvbsrc.c
+++ b/sys/dvb/gstdvbsrc.c
@@ -105,7 +105,8 @@ enum
ARG_DVBSRC_TRANSMISSION_MODE,
ARG_DVBSRC_HIERARCHY_INF,
ARG_DVBSRC_TUNE,
- ARG_DVBSRC_INVERSION
+ ARG_DVBSRC_INVERSION,
+ ARG_DVBSRC_STATS_REPORTING_INTERVAL
};
static void gst_dvbsrc_output_frontend_stats (GstDvbSrc * src);
@@ -246,9 +247,9 @@ gst_dvbsrc_inversion_get_type (void)
{
static GType dvbsrc_inversion_type = 0;
static GEnumValue inversion_types[] = {
- {INVERSION_AUTO, "AUTO", "AUTO"},
+ {INVERSION_OFF, "OFF", "OFF"},
{INVERSION_ON, "ON", "ON"},
- {INVERSION_AUTO, "OFF", "OFF"},
+ {INVERSION_AUTO, "AUTO", "AUTO"},
{0, NULL, NULL},
};
@@ -384,50 +385,57 @@ gst_dvbsrc_class_init (GstDvbSrcClass * klass)
g_object_class_install_property (gobject_class, ARG_DVBSRC_BANDWIDTH,
g_param_spec_enum ("bandwidth",
"bandwidth",
- "Bandwidth (DVB-T)", GST_TYPE_DVBSRC_BANDWIDTH, 1, G_PARAM_WRITABLE));
+ "Bandwidth (DVB-T)", GST_TYPE_DVBSRC_BANDWIDTH, 1,
+ G_PARAM_READWRITE));
g_object_class_install_property (gobject_class, ARG_DVBSRC_CODE_RATE_HP,
g_param_spec_enum ("code-rate-hp",
"code-rate-hp",
"High Priority Code Rate (DVB-T)",
- GST_TYPE_DVBSRC_CODE_RATE, 1, G_PARAM_WRITABLE));
+ GST_TYPE_DVBSRC_CODE_RATE, 1, G_PARAM_READWRITE));
g_object_class_install_property (gobject_class, ARG_DVBSRC_CODE_RATE_LP,
g_param_spec_enum ("code-rate-lp",
"code-rate-lp",
"Low Priority Code Rate (DVB-T)",
- GST_TYPE_DVBSRC_CODE_RATE, 1, G_PARAM_WRITABLE));
+ GST_TYPE_DVBSRC_CODE_RATE, 1, G_PARAM_READWRITE));
g_object_class_install_property (gobject_class, ARG_DVBSRC_GUARD,
g_param_spec_enum ("guard",
"guard",
"Guard Interval (DVB-T)",
- GST_TYPE_DVBSRC_GUARD, 1, G_PARAM_WRITABLE));
+ GST_TYPE_DVBSRC_GUARD, 1, G_PARAM_READWRITE));
g_object_class_install_property (gobject_class, ARG_DVBSRC_MODULATION,
g_param_spec_enum ("modulation",
"modulation",
"Modulation (DVB-T)",
- GST_TYPE_DVBSRC_MODULATION, 1, G_PARAM_WRITABLE));
+ GST_TYPE_DVBSRC_MODULATION, 1, G_PARAM_READWRITE));
g_object_class_install_property (gobject_class,
ARG_DVBSRC_TRANSMISSION_MODE,
g_param_spec_enum ("trans-mode",
"trans-mode",
"Transmission Mode (DVB-T)",
- GST_TYPE_DVBSRC_TRANSMISSION_MODE, 1, G_PARAM_WRITABLE));
+ GST_TYPE_DVBSRC_TRANSMISSION_MODE, 1, G_PARAM_READWRITE));
g_object_class_install_property (gobject_class, ARG_DVBSRC_HIERARCHY_INF,
g_param_spec_enum ("hierarchy",
"hierarchy",
"Hierarchy Information (DVB-T)",
- GST_TYPE_DVBSRC_HIERARCHY, 1, G_PARAM_WRITABLE));
+ GST_TYPE_DVBSRC_HIERARCHY, 1, G_PARAM_READWRITE));
g_object_class_install_property (gobject_class, ARG_DVBSRC_INVERSION,
g_param_spec_enum ("inversion",
"inversion",
"Inversion Information (DVB-T)",
- GST_TYPE_DVBSRC_INVERSION, 1, G_PARAM_WRITABLE));
+ GST_TYPE_DVBSRC_INVERSION, 1, G_PARAM_READWRITE));
+ g_object_class_install_property (gobject_class,
+ ARG_DVBSRC_STATS_REPORTING_INTERVAL,
+ g_param_spec_uint ("stats-reporting-interval",
+ "stats-reporting-interval",
+ "The number of reads before reporting frontend stats",
+ 0, G_MAXUINT, 100, G_PARAM_READWRITE));
}
/* initialize the new element
@@ -449,7 +457,7 @@ gst_dvbsrc_init (GstDvbSrc * object, GstDvbSrcClass * klass)
object->fd_dvr = -1;
for (i = 0; i < MAX_FILTERS; i++) {
- object->pids[i] = 0;
+ object->pids[i] = G_MAXUINT16;
object->fd_filters[i] = -1;
}
/* Pid 8192 on DVB gets the whole transport stream */
@@ -504,32 +512,44 @@ gst_dvbsrc_set_property (GObject * _object, guint prop_id,
break;
case ARG_DVBSRC_PIDS:
{
- int pid = 0;
- int pid_count = 0;
gchar *pid_string;
- gchar **pids;
- char **tmp;
GST_INFO_OBJECT (object, "Set Property: ARG_DVBSRC_PIDS");
pid_string = g_value_dup_string (value);
- tmp = pids = g_strsplit (pid_string, ":", MAX_FILTERS);
- while (*pids != NULL && pid_count < MAX_FILTERS) {
- pid = strtol (*pids, NULL, 0);
- if (pid > 0 && pid <= 8192) {
- GST_INFO_OBJECT (object, "Parsed Pid: %d\n", pid);
- object->pids[pid_count] = pid;
- pid_count++;
+ if (!strcmp (pid_string, "8192")) {
+ /* get the whole ts */
+ object->pids[0] = 8192;
+ } else {
+ int pid = 0;
+ int pid_count;
+ gchar **pids;
+ char **tmp;
+
+ tmp = pids = g_strsplit (pid_string, ":", MAX_FILTERS);
+
+ /* always add the PAT and CAT pids */
+ object->pids[0] = 0;
+ object->pids[1] = 1;
+
+ pid_count = 2;
+ while (*pids != NULL && pid_count < MAX_FILTERS) {
+ pid = strtol (*pids, NULL, 0);
+ if (pid > 1 && pid <= 8192) {
+ GST_INFO_OBJECT (object, "Parsed Pid: %d\n", pid);
+ object->pids[pid_count] = pid;
+ pid_count++;
+ }
+ pids++;
}
- pids++;
+
+ g_strfreev (tmp);
}
- g_strfreev (tmp);
/* if we are in playing, then set filters now */
GST_INFO_OBJECT (object, "checking if playing for setting pes filters");
if (GST_ELEMENT (object)->current_state == GST_STATE_PLAYING) {
GST_INFO_OBJECT (object, "Setting pes filters now");
gst_dvbsrc_set_pes_filters (object);
}
-
}
break;
case ARG_DVBSRC_SYM_RATE:
@@ -573,6 +593,10 @@ gst_dvbsrc_set_property (GObject * _object, guint prop_id,
g_mutex_unlock (object->tune_mutex);
}
break;
+ case ARG_DVBSRC_STATS_REPORTING_INTERVAL:
+ object->stats_interval = g_value_get_uint (value);
+ object->stats_counter = 0;
+ break;
default:
G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
}
@@ -610,6 +634,33 @@ gst_dvbsrc_get_property (GObject * _object, guint prop_id,
case ARG_DVBSRC_DISEQC_SRC:
g_value_set_int (value, object->diseqc_src);
break;
+ case ARG_DVBSRC_BANDWIDTH:
+ g_value_set_enum (value, object->bandwidth);
+ break;
+ case ARG_DVBSRC_CODE_RATE_HP:
+ g_value_set_enum (value, object->code_rate_hp);
+ break;
+ case ARG_DVBSRC_CODE_RATE_LP:
+ g_value_set_enum (value, object->code_rate_lp);
+ break;
+ case ARG_DVBSRC_GUARD:
+ g_value_set_enum (value, object->guard_interval);
+ break;
+ case ARG_DVBSRC_MODULATION:
+ g_value_set_enum (value, object->modulation);
+ break;
+ case ARG_DVBSRC_TRANSMISSION_MODE:
+ g_value_set_enum (value, object->transmission_mode);
+ break;
+ case ARG_DVBSRC_HIERARCHY_INF:
+ g_value_set_enum (value, object->hierarchy_information);
+ break;
+ case ARG_DVBSRC_INVERSION:
+ g_value_set_enum (value, object->inversion);
+ break;
+ case ARG_DVBSRC_STATS_REPORTING_INTERVAL:
+ g_value_set_uint (value, object->stats_interval);
+ break;
default:
G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
}
@@ -737,6 +788,9 @@ gst_dvbsrc_finalize (GObject * _object)
/* freeing the mutex segfaults somehow */
g_mutex_free (object->tune_mutex);
+
+ if (G_OBJECT_CLASS (parent_class)->finalize)
+ G_OBJECT_CLASS (parent_class)->finalize (_object);
}
@@ -755,8 +809,8 @@ gst_dvbsrc_finalize (GObject * _object)
* register the element factories and pad templates
* register the features
*/
-static gboolean
-plugin_init (GstPlugin * plugin)
+gboolean
+gst_dvbsrc_plugin_init (GstPlugin * plugin)
{
GST_DEBUG_CATEGORY_INIT (gstdvbsrc_debug, "dvbsrc", 0, "DVB Source Element");
@@ -770,12 +824,6 @@ plugin_init (GstPlugin * plugin)
GST_TYPE_DVBSRC);
}
-GST_PLUGIN_DEFINE (GST_VERSION_MAJOR,
- GST_VERSION_MINOR,
- "dvbsrc",
- "DVB Source", plugin_init, VERSION, "LGPL", "", "University of Paderborn");
-
-
static GstBuffer *
read_device (int fd, int adapter_number, int frontend_number, int size)
{
@@ -843,7 +891,6 @@ read_device (int fd, int adapter_number, int frontend_number, int size)
static GstFlowReturn
gst_dvbsrc_create (GstPushSrc * element, GstBuffer ** buf)
{
- static int quality_signal_rate = 0; /* FIXME: move into object struct? */
gint buffer_size;
GstFlowReturn retval = GST_FLOW_ERROR;
GstDvbSrc *object;
@@ -871,17 +918,15 @@ gst_dvbsrc_create (GstPushSrc * element, GstBuffer ** buf)
caps = gst_pad_get_caps (GST_BASE_SRC_PAD (object));
gst_buffer_set_caps (*buf, caps);
gst_caps_unref (caps);
-
- /* Every now and then signal signal quality */
- if (quality_signal_rate == 100) {
- gst_dvbsrc_output_frontend_stats (object);
- quality_signal_rate = 0;
- } else {
- quality_signal_rate++;
- }
} else {
GST_DEBUG_OBJECT (object, "Failed to read from device");
}
+
+ if (object->stats_interval != 0 &&
+ ++object->stats_counter == object->stats_interval) {
+ gst_dvbsrc_output_frontend_stats (object);
+ object->stats_counter = 0;
+ }
}
g_mutex_unlock (object->tune_mutex);
@@ -1188,7 +1233,7 @@ gst_dvbsrc_set_pes_filters (GstDvbSrc * object)
GST_INFO_OBJECT (object, "Setting PES filter");
for (i = 0; i < MAX_FILTERS; i++) {
- if (object->pids[i] == 0)
+ if (object->pids[i] == G_MAXUINT16)
break;
fd = &object->fd_filters[i];
@@ -1214,29 +1259,6 @@ gst_dvbsrc_set_pes_filters (GstDvbSrc * object)
GST_WARNING_OBJECT (object, "Error setting PES filter on %s: %s",
demux_dev, strerror (errno));
}
- /* always have PAT in the filter if we haven't used all our filter slots */
- if (object->pids[0] != 8192 && i < MAX_FILTERS) {
- /* pid 8192 means get whole ts */
- pes_filter.pid = 0;
- pes_filter.input = DMX_IN_FRONTEND;
- pes_filter.output = DMX_OUT_TS_TAP;
- pes_filter.pes_type = DMX_PES_OTHER;
- pes_filter.flags = DMX_IMMEDIATE_START;
- fd = &object->fd_filters[i];
- close (*fd);
- if ((*fd = open (demux_dev, O_RDWR)) < 0) {
- GST_WARNING_OBJECT ("Error opening demuxer: %s (%s)",
- strerror (errno), demux_dev);
- } else {
- GST_INFO_OBJECT (object, "Setting pes-filter, pid = %d, type = %d",
- pes_filter.pid, pes_filter.pes_type);
-
- if (ioctl (*fd, DMX_SET_PES_FILTER, &pes_filter) < 0)
- GST_WARNING_OBJECT (object, "Error setting PES filter on %s: %s",
- demux_dev, strerror (errno));
- }
- }
g_free (demux_dev);
-
}
diff --git a/sys/dvb/gstdvbsrc.h b/sys/dvb/gstdvbsrc.h
index 46b301c1..8e6640ed 100644
--- a/sys/dvb/gstdvbsrc.h
+++ b/sys/dvb/gstdvbsrc.h
@@ -24,7 +24,7 @@ G_BEGIN_DECLS
#define DEFAULT_BUFFER_SIZE 8192
#define DEFAULT_DISEQC_SRC -1 /* disabled */
-#define MAX_FILTERS 8
+#define MAX_FILTERS 32
/* #define's don't like whitespacey bits */
#define GST_TYPE_DVBSRC \
@@ -76,6 +76,8 @@ G_BEGIN_DECLS
int inversion;
GstDvbSrcPol pol;
+ guint stats_interval;
+ guint stats_counter;
};
struct _GstDvbSrcClass
@@ -88,6 +90,7 @@ G_BEGIN_DECLS
GType gst_dvbsrc_get_type (void);
+ gboolean gst_dvbsrc_plugin_init (GstPlugin *plugin);
G_END_DECLS