【发布时间】:2016-11-02 22:30:47
【问题描述】:
在大型机上用 C 语言编写的供应商软件包提供了通过用户出口覆盖其部分功能的选项。提供的 C 函数原型为:
extern int SomeExit (void * Parameters,
void * Record1,
void * Record2,
char ComparisonType,
char * RankString,
void * NotUsed1,
int * NotUsed2)
由于我们主要是一家 COBOL 商店,因此我定义了一个 Enterprise COBOL 4.2 程序(作为 DLL)来实现出口,试图遵循 IBM ILC 指南 (https://www.ibm.com/support/knowledgecenter/en/SSLTBW_1.13.0/com.ibm.zos.r13.ceea400/clcccb5.htm#clcccb5) 中给出的约定以及这个旧 SHARE 演示文稿中的示例:http://www-01.ibm.com/support/docview.wss?uid=swg27003846&aid=1 但生成的程序在调用退出时异常终止,并且在我的DISPLAY 消息之前。我的假设是我没有正确声明接收数据结构。下面是我当前测试代码的 sn-p (忽略我的命名约定 - 这是一个原型,用于证明接口,一旦我有基本的调用工作,将重写为我们的内部标准)。
IDENTIFICATION DIVISION.
PROGRAM-ID. "SomeExit".
...
LINKAGE SECTION.
01 WS-PARAMETERS-POINTER USAGE IS POINTER SYNCHRONIZED.
01 SORT-PASS-RECORD1-POINTER USAGE IS POINTER SYNCHRONIZED.
01 SORT-PASS-RECORD2-POINTER USAGE IS POINTER SYNCHRONIZED.
01 WS-COMPARISION-TYPE PIC X.
01 WS-RANK-STRING-POINTER USAGE IS POINTER SYNCHRONIZED.
01 WS-NOT-USED1-POINTER USAGE IS POINTER SYNCHRONIZED.
01 WS-NOT-USED2-POINTER USAGE IS POINTER SYNCHRONIZED.
01 WS-RETURN PIC S9(9) USAGE IS BINARY.
...
PROCEDURE DIVISION USING BY VALUE WS-PARAMETERS-POINTER
SORT-PASS-RECORD1-POINTER
SORT-PASS-RECORD2-POINTER
WS-COMPARISION-TYPE
WS-RANK-STRING-POINTER
WS-NOT-USED1-POINTER
WS-NOT-USED2-POINTER
RETURNING WS-RETURN.
DISPLAY 'IN EXIT'.
...
MOVE 0 TO WS-RETURN.
GOBACK.
异常结束是:
CEE3250C The system or user abend U 016 R=00000000 was issued.
From entry point main at compile unit offset +00000192 at entry offset +00000192 at address 28500ECA.
供应商代码动态调用 DLL。当我删除 DLL 时,我收到一条消息,指出找不到出口,因此 C 代码似乎正在尝试调用它。
我尝试了PROCEDURE DIVISION USING 的变体,包括删除BY VALUE,使用BY REFERENCE(虽然我知道这是默认设置)并用实际的结构定义替换POINTERs。我是否误解了有关如何构造传递给 COBOL 例程的参数的手册?
编辑:我确实向供应商开放了支持票,但他们尚未回复任何有用的信息。
谢谢,大卫
【问题讨论】:
-
什么异常?什么程序中的什么位移?为什么
BY VALUE对于所有内容,我只考虑比较类型,其他通过引用?你知道那是做什么的吗?忘记SYNCHRONIZED。为什么为您的 LINKAGE SECTION 项目添加 WS- 前缀?严格来说,您应该始终在 GOBACK 之前将 WS-RETURN 设置为某个值。哪个版本的企业 COBOL?来自 C 的调用是否来自 DLL?您是否询问过供应商(您的许可包括支持)。 -
一小时内查看了 23 次,获得了 2 次点赞。这是一个哇!对于 COBOL 问题。可能意味着比 COBOL 人阅读更多的 C,比 Mainframe 更多的非 Mainframe :-)
-
你的
PROGRAM-ID是什么?出口是静态链接还是供应商例程fetching 你的 cobol-dll? -
谢谢。
From entry point main表示异常结束来自 C(或 C++)程序,因为 Enterprise COBOL 没有“main” - 如果它是您的程序中的异常结束,则此时会显示 SomeExit。它是用户异常终止,代码为 16。用户异常终止是特定于应用程序的。您正在运行的产品已生成异常结束。这意味着它已将某些内容识别为不正确,并且该不正确是最终的。检查所有假脱机文件(从第三个开始),因为可能有(也可能没有)附加消息。 -
因为你的模块根本不存在,你会失败(尽管哪个异常结束代码?S806?)@piet.t 关于出口是否应该是静态链接的观点值得确认在退出的文档中。对于 POINTER 项目,请使用 BY REFERENCE。另一方面,您可以使用 BY VALUE。我认为您可以将 RETURNING 项目保持原样。我无法想象那里的任何变化都会影响异常结束。您知道供应商代码是否为 64 位吗?如果是,请忘记使用 COBOL 程序。#