简体   繁体   English

如何将小部件作为 client_data 传递给 Xt/Motif 信号处理程序?

[英]How do I pass a widget as client_data to a Xt/Motif signal handler?

I want a Motif application to redraw a Drawing Area widget on reception of SIGUSR1 signal.我想要一个 Motif 应用程序在接收到 SIGUSR1 信号时重绘一个绘图区小部件。

I have configured the signal using the Xt functionalities in X11R6:我已经使用 X11R6 中的 Xt 功能配置了信号:

/* CONFIGURE READ SIGNAL, TRAPS SIGUSR1 TO FORCE APPLICATION READ DATA FILE */
signal(SIGUSR1, signal_usr1_handler);
signal_id = XtAppAddSignal (app, read_data, (XtPointer)chart_area);

Where signal_usr1_handler is the OS Unix handler and read_data is the Xt signal handler.其中signal_usr1_handler是 OS Unix 处理程序,而read_data是 Xt 信号处理程序。

For the Xt signal handler I am trying to pass the widget chart_area which is a Drawing Area Widget.对于 Xt 信号处理程序,我试图传递小部件chart_area ,它是一个绘图区小部件。

These are the handlers:这些是处理程序:

/* SIGNAL HANDLER */
void signal_usr1_handler() {
    printf("SIGUSR1 RECEIVED\n");
    XtNoticeSignal(signal_id);
}
    
/* READ DATA XT SIGNAL HANDLER */
void read_data(XtPointer client_data, XtSignalId *id) {

    posx += 5;
    printf("XT HANDLES SIGUSR1\n"); 
    
    XmDrawingAreaCallbackStruct da_struct;

    da_struct.reason = XmCR_EXPOSE;
    da_struct.event = (XEvent *) NULL;
    da_struct.window = XtWindow((Widget)client_data);

    XtCallCallbacks((Widget)client_data, XmNexposeCallback, (XtPointer) &da_struct);
    
}

The application raises a Segmentation fault (core dumped) exactly when executing the XtCallCallbacks function.应用程序恰好在执行XtCallCallbacks函数时引发Segmentation fault (core dumped)

This is the complete source code:这是完整的源代码:

#include <stdlib.h>
#include <stdio.h>
#include <signal.h>
#include <sys/wait.h>
 
#include <Xm/Xm.h>
#include <Xm/DrawingA.h>
#include <Xm/Label.h>

/* FUNCTION DECLARATIONS */
void draw_chart(Widget widget, XtPointer client_data, XtPointer call_data);
void signal_usr1_handler();
void read_data(XtPointer client_data, XtSignalId *id);

/* XT SIGNAL ID */
XtSignalId  signal_id;      /* Signal ID used to receive data read via SIGUSR1 */
     
int main(int argc, char *argv[]) {

    /* WIDGETS */
    XtAppContext    app;            /* Application Context */
    Widget      toplevel;       /* Top Level Button */
    Widget      chart_area;     /* Drawing Area Widget to draw the chart */
        
    /* DRAWING AREA RELATED */
    XGCValues   gcv;
    GC      gc;
    
    /* RESOURCE VALUE ARRAYS/COUNT */
    Arg al[10];
    int ac;
    
    /* INITIALIZE TOP LEVEL WINDOW */
    XtSetLanguageProc(NULL, NULL, NULL);
    toplevel = XtVaOpenApplication( 
        &app, argv[0], NULL, 0, &argc, argv, NULL, sessionShellWidgetClass,
        XmNwidth, 400, XmNheight, 300, NULL
    );
    
    /* CREATE AND MANAGE DRAWING CANVAS WIDGET */
    ac=0; 
    chart_area = XmCreateDrawingArea(toplevel, "chart_area", al, ac);
                
    /* CREATE GRAPHICS CONTEXT */
    gcv.foreground = WhitePixelOfScreen(XtScreen(chart_area));
    gcv.background = BlackPixelOfScreen(XtScreen(chart_area));  
    gc = XCreateGC (
        XtDisplay(chart_area),
        RootWindowOfScreen(XtScreen(chart_area)), 
        (GCForeground | GCBackground), 
        &gcv);

    /* ASSIGN GRAPHICS CONTEXT */   
    XtVaSetValues(chart_area, XmNuserData, gc, NULL);   
    
    /* ASSIGN CALLBACKS AND MANAGE WIDGET */
    XtAddCallback(chart_area, XmNexposeCallback, draw_chart, NULL);
    XtManageChild(chart_area);  
                    
    /* CONFIGURE READ SIGNAL, TRAPS SIGUSR1 TO FORCE APPLICATION READ DATA FILE */
    signal(SIGUSR1, signal_usr1_handler);
    signal_id = XtAppAddSignal (app, read_data, (XtPointer)chart_area);
        
    /* REALIZE TOPLEVEL WINDOW AND LAUNCH APPLICATION LOOP */
    XtRealizeWidget(toplevel);  
    XtAppMainLoop(app);
    
    return 0;

}

void draw_chart(Widget widget, XtPointer client_data, XtPointer call_data) {

    XmDrawingAreaCallbackStruct *cbs = (XmDrawingAreaCallbackStruct *) call_data;
    XEvent *event = cbs->event;
    Display *dpy = event->xany.display;
    GC gc;
    
    /* DRAW LINE */
    if(cbs->reason == XmCR_EXPOSE || cbs->reason == XmCR_ACTIVATE) {

        XtVaGetValues(widget, XmNuserData, &gc, NULL);
        XDrawLine(dpy, cbs->window, gc, 10, 10, posx, 200);

    }
}
    
/* SIGNAL HANDLER */
void signal_usr1_handler() {
    printf("SIGUSR1 RECEIVED\n");
    XtNoticeSignal(signal_id);
}
    
/* READ DATA XT SIGNAL HANDLER */
void read_data(XtPointer client_data, XtSignalId *id) {

    /* ... READ DATA AND PROCESSING GOES HERE ... */

    posx += 5;
    printf("XT HANDLES SIGUSR1\n"); 
    
    XmDrawingAreaCallbackStruct da_struct;

    da_struct.reason = XmCR_EXPOSE;
    da_struct.event = (XEvent *) NULL;
    da_struct.window = XtWindow((Widget)client_data);

    /* XtCallCallbacks((Widget)client_data, XmNexposeCallback, (XtPointer) &da_struct); */
    
}

The following causes the crash:以下原因导致崩溃:

Display *dpy = event->xany.display;显示 *dpy = 事件->xany.display;

Because event was set to NULL.因为事件被设置为NULL。

Also you may need to deal with gc by passing its address and receiving XmUserData as此外,您可能需要通过传递其地址并接收 XmUserData 来处理 gc

GC *gc; GC*gc;

then using *gc,然后使用*gc,

声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.

 
粤ICP备18138465号  © 2020-2024 STACKOOM.COM