Передача целого числа, хранящегося в mmap, в GTK - PullRequest
1 голос
/ 17 апреля 2020

РЕДАКТИРОВАТЬ

Этот вопрос оказался двумя проблемами, объединенными в одну. Тем не менее, я не могу удалить вопрос. Объем первоначального вопроса об указателях был решен Дэвидом Раньери. Проблема mmap / fork / gtk будет предметом нового вопроса и не будет рассматриваться здесь.


Я хочу напечатать значение, которое я сохранил в памяти в окне GTK. Целое число должно храниться с использованием mmap, чтобы его можно было сохранить во время разветвления в другом месте кода. Я не могу сослаться на этот адрес, отображенный в памяти из GTK, не получив SegFault. Я что-то здесь не так делаю? Есть ли лучший способ?

Моя текущая стратегия:

  • Резервная память для int с mmap при *VAL
  • Процесс Fork, один половина изменяет VAL, другая половина запускает GTK
  • Передача *VAL приложению в слот пользовательских данных
  • Пользовательские данные теперь называются localval в activate()
  • Распечатать значение по адресу localval путем преобразования из gpointer в int.

MWE (это вызывает segfault, работайте на свой страх и риск):

/*
 * MMAP Variable SegFault
 */

#include <stdlib.h>
#include <unistd.h>
#include <sys/mman.h>
#include <gtk/gtk.h>

static void activate (GtkApplication *app, gpointer *localval) {
    GtkWidget *window;
    // Button Containers
    GtkWidget *button_box_quit;
    // Buttons
    GtkWidget *exit_button;
    // Text
    GtkWidget *text_status;

    // Define Window, dynamic size for screen.
    window = gtk_application_window_new (app);
    gtk_window_set_title (GTK_WINDOW (window), "test");
    gtk_window_set_default_size(GTK_WINDOW(window), 400, 300);

    // Define Button Boxes.
    button_box_quit = gtk_button_box_new(GTK_ORIENTATION_HORIZONTAL);

    // Define Exit Button, put it in a box, put box in window
    exit_button = gtk_button_new_with_label ("Exit");
    gtk_container_add(GTK_CONTAINER (button_box_quit), exit_button);
    gtk_container_add(GTK_CONTAINER (window), button_box_quit);

    // Connect signals to buttons
    g_signal_connect_swapped (exit_button, "clicked", G_CALLBACK (gtk_widget_destroy), window);

    // Define text status
    char msg[32]={0};
    // The "print" line
    g_snprintf(msg, sizeof msg, "val: %d\n", GPOINTER_TO_INT(*localval));
    text_status = gtk_label_new(msg);
    gtk_container_add(GTK_CONTAINER (button_box_quit), text_status);

    //Activate!
    gtk_widget_show_all (window);
}

int main (int argc, char **argv) {
    GtkApplication *app;
    int status;

    int *VAL = mmap(NULL, sizeof(int), PROT_READ|PROT_WRITE, MAP_SHARED|MAP_ANONYMOUS, -1, 0);
    int *ABORT = mmap(NULL, sizeof(int), PROT_READ|PROT_WRITE, MAP_SHARED|MAP_ANONYMOUS, -1, 0);
    int pid = fork();

    if (pid == 0) {
        while(!*ABORT) {
            printf("%d\n", *VAL);
            *VAL = *VAL + 1;
            usleep(1000000);
        }
        exit(0);
    } else {
        app = gtk_application_new ("org.gtk.example", G_APPLICATION_FLAGS_NONE);
        // The passing line
        g_signal_connect (app, "activate", G_CALLBACK (activate), (gpointer *)*VAL);
        status = g_application_run (G_APPLICATION (app), argc, argv);
        g_object_unref (app);
        *ABORT = 1;
    }
    *ABORT = 1;
    return status;
}


В нерабочей альтернативе этому коду я попытался изменить строку печати на:

g_snprintf(msg, sizeof msg, "val: %d\n", &GPOINTER_TO_INT(*localval));

Но компилятор считает, что я хочу использовать & в качестве компаратора.

1 Ответ

1 голос
/ 17 апреля 2020

Вы передаете разыменованный указатель (значение) функции, ожидающей указатель:

int *VAL = ...;
...
g_signal_connect (app, "activate", G_CALLBACK (activate), (gpointer *)*VAL);

также переключитесь на

g_signal_connect (app, "activate", G_CALLBACK (activate), VAL); // Do not use a wrong cast (void **)

, gpointer является псевдонимом void *, используя gpointer *data, вы получаете void **data, а не то, что вам нужно, поэтому

static void activate (GtkApplication *app, gpointer *localval) {

должно быть

static void activate (GtkApplication *app, gpointer localval) { // without *

, наконец, чтобы напечатать значение указателя, используйте

g_snprintf(msg, sizeof msg, "val: %d\n", *(int *)localval); 
...