Search code examples
cgtkgtk3

Segmentation fault when using gtk_threads_add_timeout


I have created a very simple gtk interface that displays the time. As such it will need to be updated frequently and after looking into the docs, gtk_threads_add_timeout seemed a logical method to facilitate this. I adapted the function "set_time" to work with the gtk function and ended up with the following:

#include <stdlib.h>
#include <stdio.h>
#include <string.h>
#include <unistd.h>
#include <fcntl.h>
#include <time.h>

#include <gtk/gtk.h>

typedef struct
{
  GtkWidget *label;
} time_args;

static int set_time(void *args);

int main (int argc, char *argv[])
{
    GtkBuilder *builder;
    GtkWidget *window;
    GtkLabel *time;
    time_args t_args;

    gtk_init(&argc, &argv);

    builder = gtk_builder_new();
    gtk_builder_add_from_file (builder, temp, NULL);

    window = GTK_WIDGET(gtk_builder_get_object(builder, "window_main"));
    gtk_builder_connect_signals(builder, NULL);
    gtk_window_fullscreen(GTK_WINDOW(gtk_widget_get_toplevel(window)));

    time = GTK_LABEL(gtk_builder_get_object(builder, "time"));
    t_args.label = GTK_WIDGET(time);
    set_time(&t_args);
    gdk_threads_add_timeout(1000, set_time, &t_args);

    g_object_unref(builder);

    gtk_widget_show(window);
    gtk_main();

    return 0;
}

void on_window_main_destroy()
{
    gtk_main_quit();
}

static int set_time(void *args) {
    time_args *data = (time_args *)args;
    time_t rawtime = time(&rawtime);
    struct tm *timeinfo = localtime(&rawtime);
    char *timeout;
    int hour = timeinfo->tm_hour;

    if(hour > 12) {
        hour -= 12;
        sprintf(timeout, "%d:%02d %s", hour, timeinfo->tm_min, "PM");
    } else {
        sprintf(timeout, "%d:%02d %s", hour, timeinfo->tm_min, "AM");
    }

    gtk_label_set_text(GTK_LABEL(data->label), timeout);

    return G_SOURCE_CONTINUE;
}

It compiles fine with no warnings and calling set_time(&t_args) works fine on the first call, at the first instance of gdk_threads_add_timeout(1000, set_time, &t_args) however I get a segmentation fault. Anyone have any similar issues or have a similar use case that was solved using a different method?


Solution

  • You need to reserve space for timeout before using sprintf:

    In this line:

    sprintf(timeout, "%d:%02d %s", hour, timeinfo->tm_min, "PM");
    

    The size of the buffer should be large enough to contain the entire resulting string, but you are using an uninitialized char *

    char *timeout;
    

    shoulde be

    char timeout[32]; // Some arbitrary size