我最近发现了曼弗雷德·马洛四年前的一篇文章,这篇文章差一点就解决了我遇到的一个问题。本文“使用Glade在Forth中创建GTK+应用程序”讨论了信号处理程序,指出不能使用Forth,而只能使用C创建它们。我认为我的问题仍然是一个小问题,但我很困惑,我会非常感激任何建议。其次,如果有人知道如何在地址不使用破折号的情况下给Mahlow先生发一封电子邮件,我很想听听。
我正在使用Linux版本的SwiftForth编写一个应用程序,以创建一个自定义窗口,并接收按键并将其写入该窗口。我的目标是实现查尔斯摩尔的colorForth,增强和更友好的用户。我决定使用GTK+,因为我对它知之甚少,但我成功地使用了回调(信号处理程序)。整个过程中只有两页代码,所以我把它放在这里:
\ ------------------------ COLOR FORTH -----------------------
ANEW TASK-COLORFORTH
LIBRARY /usr/lib/i386-linux-gnu/libgtk-3.so.0.1000.8
FUNCTION: gtk_window_new ( code -- addr )
FUNCTION: gtk_widget_destroy ( wptr -- )
FUNCTION: gtk_window_close ( wptr -- )
FUNCTION: gtk_widget_show ( wptr -- )
FUNCTION: gtk_init ( -- )
FUNCTION: gtk_window_set_decorated ( wptr flag -- )
FUNCTION: gtk_window_move ( wptr x y -- )
\ FUNCTION: gtk_widget_new ( n addr -- ) ( try 0 0 )
\ FUNCTION: gtk_widget_map ( wptr -- )
\ FUNCTION: gtk_widget_show_all ( wptr -- )
FUNCTION: gtk_window_get_screen ( wptr -- wgptr )
\ FUNCTION: gtk_window_present ( wptr -- )
FUNCTION: gtk_window_set_title ( wptr TitlePtr -- )
FUNCTION: gtk_window_resize ( wptr width height -- )
FUNCTION: gtk_widget_modify_bg ( wptr state cptr -- )
\ FUNCTION: gtk_widget_modify_fg ( wptr state cptr -- )
FUNCTION: gtk_widget_modify_text ( wptr state cptr -- )
FUNCTION: gtk_widget_modify_base ( wptr statre cptr -- )
FUNCTION: g_signal_connect_data ( wptr name callback data 0 0 -- )
FUNCTION: gtk_main ( -- )
FUNCTION: gtk_main_quit ( -- )
FUNCTION: gtk_widget_set_events ( wptr flag -- )
LIBRARY /usr/lib/i386-linux-gnu/libgdk-3.so.0.1000.8
FUNCTION: gdk_event_get_keyval ( eptr kptr -- )
FUNCTION: gdk_event_get_state ( eptr sptr -- )
FUNCTION: gdk_event_get_event_type ( eptr -- type ) ( requires GET.RETURN )
\ LIBRARY /usr/lib/i386-linux-gnu/libgio-2.0.so.0.4000.0
\ LIBRARY /lib/i386-linux-gnu/libglib-2.0.so.0.4000.0
\ LIBRARY /usr/lib/i386-linux-gnu/libpango-1.0.so.0.3600.3
\ LIBRARY /usr/lib/i386-linux-gnu/libatk-1.0.so.0.21009.1
\ LIBRARY /usr/lib/i386-linux-gnu/libgobject-2.0.so.0.4000.0
\ LIBRARY /usr/lib/i386-linux-gnu/libgdk_pixbuf-2.0.so.0.3000.7
VARIABLE Window-Ptr
VARIABLE Event-Ptr
VARIABLE KeyVal
VARIABLE KeyState
Z" ColorForth" VALUE Title
Z" key_press_event" VALUE Keypress-Name
Z" delete_event" VALUE Delete-Event-Name
Z" destroy" VALUE Destroy-Name
CREATE Background 0 , 0 H, 0 H, $2000 H,
CREATE Black 0 , 0 H, 0 H, 0 H,
CREATE White 0 , $E000 H, $E000 H, $8000 H,
\ ----------------------------------------------------------------
ICODE LEAVE.FALSE
EAX EAX SUB
RET
END-CODE
ICODE GET.RETURN
4 # EBP SUB
EBX 0 [EBP] MOV
EAX EBX MOV
RET
END-CODE
\ This may be the part where I am having problems
: SHUTDOWN
Window-Ptr @ DUP gtk_window_close
gtk_widget_destroy
;
: CB.DELETE.EVENT ( wptr data -- false to destroy )
RDROP RDROP ( assuming 2 paramaters on Rstack and return -1 )
LEAVE.FALSE
;
: CB.DESTROY ( wptr data -- )
RDROP RDROP ( two parameters here )
gtk_main_quit
;
: CB.KEYPRESS ( wptr eptr data -- ) ( assuming 3 parameters on Rstack )
Window-Ptr @ . R> . R> . R> . R@ ( This for diagnosis – eventually )
SHUTDOWN ( goes to a [SWITCH )
;
: STARTUP ( -- )
Window-Ptr OFF Event-Ptr OFF
gtk_init
0 gtk_window_new ?DUP ( 0 = GTK_WINDOW_TOPLEVEL )
IF DUP Window-Ptr !
DUP 1024 gtk_widget_set_events ( 1024 for Keypress signals )
DUP Title gtk_window_set_title
DUP 1280 850 gtk_window_resize
DUP 0 gtk_window_set_decorated
DUP 0 Background gtk_widget_modify_bg
DUP Delete-Event-Name [ ' CB.DELETE.EVENT +ORIGIN ] LITERAL 0
0 0 g_signal_connect_data
DUP Destroy-Name [ ' CB.DESTROY +ORIGIN ] LITERAL 0
0 0 g_signal_connect_data
DUP Keypress-Name [ ' CB.KEYPRESS +ORIGIN ] LITERAL 0
0 0 g_signal_connect_data
gtk_widget_show
THEN
gtk_main
;我无法使用通常的g_signal_connect,因为我在任何库中都找不到它。如果它出现了,我会使用它,因为代码更简单。但是,上面的代码会编译,当我输入启动时,它就是这样做的。正如我所指定的,窗口就在那里,信号被设置,但是按下带有“分段错误”的任何键崩溃SwiftForth。
正如您所看到的,我将处理程序的绝对地址提供给信号连接函数。我还尝试将该地址放入变量并将其作为指针传递,但这将产生一个GTK+错误。
在编写处理程序时,我假设它会像任何库调用一样运行。输入参数将在返回堆栈上,在执行处理程序代码时由GTK提供,我应该将任何返回放入EAX中。但这种分析可能是错误的。我可能准备的地址错误或处理程序的参数错误,或有完全错误的概念。
我希望,尽管必须在C中使用处理程序,但在必要时可以在汇编程序中模拟C代码所做的事情。
我感觉到我离取得这一突破还有一段路要走,如果有人能给我一些建议或我可以去看的地方,我将非常感激。
发布于 2014-12-15 19:45:14
FUNCTION: g_signal_connect_data ( wptr name callback data 0 0 -- ),我看到您正在使用像' CB.DELETE.EVENT +ORIGIN这样的东西作为回调参数传递。
你不应该首先使用Forth回调吗?您的CB.DELETE.EVENT (和其他部分)需要并期望有第四个上下文。
在SwiftForth中,这可以通过
' CB.DELETE.EVENT 2 CB: *CB.DELETE>EVENT将*CB.DELETE>EVENT作为回调参数传递给g_signal_connect_data
CB:为CB.DELETE.EVENT创建了一个包装器,它建立了运行在其中的临时Forth环境。有关访问给回调的参数,请参阅SwiftForth手册。当然,其他回调也应该进行调整。
注意: SwiftForth以特定的方式处理输入。其他四种系统在处理回调输入参数方面有所不同。但是一般来说,不要盲目地依赖返回堆栈是很好的。始终检查第四文件。
导入库单词可以在Forth环境中运行非Forth函数,而在外部上下文中运行Forth words的反向操作则是用回调词完成的。它们允许信号处理程序、完成处理程序、异常处理程序、POSIX线程的启动例程、Objective类的方法实现等。许多在操作系统下运行的Forth系统(如SwiftForth、VFX、iForth、MacForth、iMops和GF4)提供了创建Forth回调的方法。所以我想这里不能用的任何说法都被驳斥了。见这个第四+ GTK示例。
顺便说一下,我在SwiftForth OS X.OK ;-)中运行了经过修改的代码。)
https://stackoverflow.com/questions/26557963
复制相似问题