summaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
-rw-r--r--src/cameras/uca-mock-camera.c8
-rw-r--r--test/CMakeLists.txt7
-rw-r--r--test/perf-overhead.c143
3 files changed, 154 insertions, 4 deletions
diff --git a/src/cameras/uca-mock-camera.c b/src/cameras/uca-mock-camera.c
index e885475..0198b9f 100644
--- a/src/cameras/uca-mock-camera.c
+++ b/src/cameras/uca-mock-camera.c
@@ -68,7 +68,7 @@ struct _UcaMockCameraPrivate {
guint height;
gfloat frame_rate;
gfloat max_frame_rate;
- guint16 *dummy_data;
+ guint8 *dummy_data;
gboolean thread_running;
@@ -100,7 +100,7 @@ static gpointer mock_grab_func(gpointer data)
const gulong sleep_time = (gulong) G_USEC_PER_SEC / priv->frame_rate;
while (priv->thread_running) {
- camera->grab_func(NULL, camera->user_data);
+ camera->grab_func(priv->dummy_data, camera->user_data);
g_usleep(sleep_time);
}
@@ -282,8 +282,8 @@ static void uca_mock_camera_init(UcaMockCamera *self)
self->priv = UCA_MOCK_CAMERA_GET_PRIVATE(self);
self->priv->width = 640;
self->priv->height = 480;
- self->priv->frame_rate = self->priv->max_frame_rate = 100.0f;
- self->priv->dummy_data = (guint16 *) g_malloc0(self->priv->width * self->priv->height);
+ self->priv->frame_rate = self->priv->max_frame_rate = 100000.0f;
+ self->priv->dummy_data = (guint8 *) g_malloc0(self->priv->width * self->priv->height);
self->priv->grab_thread = NULL;
self->priv->binnings = g_value_array_new(1);
diff --git a/test/CMakeLists.txt b/test/CMakeLists.txt
index 50aa2b8..096c287 100644
--- a/test/CMakeLists.txt
+++ b/test/CMakeLists.txt
@@ -65,3 +65,10 @@ target_link_libraries(test-all
${GLIB2_LIBRARIES}
${GOBJECT2_LIBRARIES}
)
+
+add_executable(perf-overhead perf-overhead.c)
+target_link_libraries(perf-overhead
+ uca-gobject
+ ${GLIB2_LIBRARIES}
+ ${GOBJECT2_LIBRARIES}
+ )
diff --git a/test/perf-overhead.c b/test/perf-overhead.c
new file mode 100644
index 0000000..20ca6c3
--- /dev/null
+++ b/test/perf-overhead.c
@@ -0,0 +1,143 @@
+/* Copyright (C) 2011, 2012 Matthias Vogelgesang <matthias.vogelgesang@kit.edu>
+ (Karlsruhe Institute of Technology)
+
+ This library is free software; you can redistribute it and/or modify it
+ under the terms of the GNU Lesser General Public License as published by the
+ Free Software Foundation; either version 2.1 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 Lesser General Public License for more
+ details.
+
+ You should have received a copy of the GNU Lesser General Public License along
+ with this library; if not, write to the Free Software Foundation, Inc., 51
+ Franklin St, Fifth Floor, Boston, MA 02110, USA */
+
+#include <glib-object.h>
+#include <signal.h>
+#include <stdio.h>
+#include <stdlib.h>
+#include <string.h>
+#include "uca-camera.h"
+
+#define handle_error(errno) {if ((errno) != UCA_NO_ERROR) printf("error at <%s:%i>\n", \
+ __FILE__, __LINE__);}
+
+typedef struct {
+ guint counter;
+ gsize size;
+ gpointer destination;
+} thread_data;
+
+static UcaCamera *camera = NULL;
+
+static void sigint_handler(int signal)
+{
+ printf("Closing down libuca\n");
+ uca_camera_stop_recording(camera, NULL);
+ g_object_unref(camera);
+ exit(signal);
+}
+
+static void test_synchronous_operation(UcaCamera *camera)
+{
+ GError *error = NULL;
+ guint width, height, bits;
+ g_object_get(G_OBJECT(camera),
+ "sensor-width", &width,
+ "sensor-height", &height,
+ "sensor-bitdepth", &bits,
+ NULL);
+
+ const int pixel_size = bits == 8 ? 1 : 2;
+ const gsize size = width * height * pixel_size;
+ const guint n_trials = 10000;
+ gpointer buffer = g_malloc0(size);
+
+ uca_camera_start_recording(camera, &error);
+ GTimer *timer = g_timer_new();
+
+ for (guint n = 0; n < n_trials; n++)
+ uca_camera_grab(camera, &buffer, &error);
+
+ gdouble total_time = g_timer_elapsed(timer, NULL);
+ g_timer_stop(timer);
+
+ g_print("Synchronous data transfer\n");
+ g_print(" Bandwidth: %3.2f MB/s\n", size * n_trials / 1024. / 1024. / total_time);
+ g_print(" Throughput: %3.2f frames/s\n", n_trials / total_time);
+
+ uca_camera_stop_recording(camera, &error);
+ g_free(buffer);
+ g_timer_destroy(timer);
+}
+
+static void grab_func(gpointer data, gpointer user_data)
+{
+ static GStaticMutex mutex = G_STATIC_MUTEX_INIT;
+
+ thread_data *d = (thread_data *) user_data;
+ g_memmove(d->destination, data, d->size);
+ g_static_mutex_lock(&mutex);
+ d->counter++;
+ g_static_mutex_unlock(&mutex);
+}
+
+static void test_asynchronous_operation(UcaCamera *camera)
+{
+ GError *error = NULL;
+ guint width, height, bits;
+
+ g_object_get(G_OBJECT(camera),
+ "sensor-width", &width,
+ "sensor-height", &height,
+ "sensor-bitdepth", &bits,
+ NULL);
+
+ const guint pixel_size = bits == 8 ? 1 : 2;
+
+ thread_data d = {
+ .counter = 0,
+ .size = width * height * pixel_size,
+ .destination = g_malloc0(width * height * pixel_size)
+ };
+
+ g_object_set(G_OBJECT(camera),
+ "transfer-asynchronously", TRUE,
+ NULL);
+
+ uca_camera_set_grab_func(camera, &grab_func, &d);
+ uca_camera_start_recording(camera, &error);
+ g_usleep(G_USEC_PER_SEC);
+ uca_camera_stop_recording(camera, &error);
+
+ g_print("Asynchronous data transfer\n");
+ g_print(" Bandwidth: %3.2f MB/s\n", d.size * d.counter / 1024. / 1024.);
+ g_print(" Throughput: %i frames/s\n", d.counter);
+
+ g_free(d.destination);
+}
+
+int main(int argc, char *argv[])
+{
+ GError *error = NULL;
+ (void) signal(SIGINT, sigint_handler);
+
+ g_type_init();
+ camera = uca_camera_new("mock", &error);
+
+ if (camera == NULL) {
+ g_print("Couldn't initialize camera\n");
+ return 1;
+ }
+
+ test_synchronous_operation(camera);
+ g_print("\n");
+ test_asynchronous_operation(camera);
+
+ g_object_unref(camera);
+
+ return error != NULL ? 1 : 0;
+}