odbcprepare给出致命错误:允许的内存大小已用完



我有一个Debian服务器(64位),我想通过PHP将它连接到AS400的数据库。我已经安装了IBM I Access for Linux和unixodbc。我遵循了本教程:https://www.albertopicado.es/conexion-odbc-con-base-de-datos-db2-en-un-servidor-ibm-iseries/虽然是西班牙语,但你可以看到我遵循的流程。问题是,我可以建立一个简单的连接,比如:

$server= 'DRIVER={DRIVER_NAME};DATABASE=DATABASENAME;SYSTEM=IP;HOSTNAME=IP;PORT=NUMBER_OF_THE_PORT;PROTOCOL=TCPIP;';
$as400= odbc_connect($server, "username", "password");
$as400 ? echo "ok" : echo "ko";

这个返回"ok",所以我知道连接已经建立。然后,我做了一个简单的odbcprepare,如下所示:

$query="SELECT * FROM DATABASE.TABLE WHERE ID=1;
$result=odbc_prepare($as400,$query);

这给我带来了致命错误:允许的1048576000字节的内存大小已用完(试图分配140707423584261字节)。

我已经将php.ini文件中memory_limit的值增加到1000M,但它仍然会抛出相同的错误。我一直在网上冲浪,我发现人们对64位版本有问题,比如这篇文章:Linux odbc致命错误:允许内存大小,但我正在执行的查询没有任何空值。

信息

odbcinst-j命令:

DRIVERS............: /usr/local/etc/odbcinst.ini
SYSTEM DATA SOURCES: /usr/local/etc/odbc.ini
FILE DATA SOURCES..: /usr/local/etc/ODBCDataSources
USER DATA SOURCES..: /root/.odbc.ini
SQLULEN Size.......: 8
SQLLEN Size........: 8
SQLSETPOSIROW Size.: 8

odbcinst.ini:

[iSeries Access ODBC Driver]
Description=iSeries Access for Linux ODBC Driver
Driver=/opt/ibm/iSeriesAccess/lib64/libcwbodbc.so
Setup=/opt/ibm/iSeriesAccess/lib64/libcwbodbcs.so
NOTE1=If using unixODBC 2.2.11 or later and you want the 32 and 64-bit ODBC drivers to share DSN's,
NOTE2=the following Driver64/Setup64 keywords will provide that support.
Driver64=/opt/ibm/iSeriesAccess/lib64/libcwbodbc.so
Setup64=/opt/ibm/iSeriesAccess/lib64/libcwbodbcs.so
Threading=2
DontDLClose=1
UsageCount=2
[iSeries Access ODBC Driver 64-bit]
Description=iSeries Access for Linux 64-bit ODBC Driver
Driver=/opt/ibm/iSeriesAccess/lib64/libcwbodbc.so
Setup=/opt/ibm/iSeriesAccess/lib64/libcwbodbcs.so
Threading=2
DontDLClose=1
UsageCount=1

odbc.ini为空。

编辑

看到可能iSeriesAccess驱动程序x64有一些错误,我终于卸载了它。然后我安装了iSeriesAccessDriverx32。在安装新的驱动程序之前,我制作了我的Debian multiarch,并下载了ia32lib(因为iSeriesAccess driver x32缺少一些软件包)。

使用新的驱动程序,我得到了另一个错误:
警告:odbc_connect():SQL错误:[unixODBC][driver Manager]无法打开lib'/opt/ibm/iSeriesAccess/lib/libcwbodbc.so':在SQLConnect中找不到文件,SQL状态为01000。

我已经仔细检查了库"libcwbodbc.so"是否存在。我还对lib运行命令ldd,看看是否缺少什么:

linux-gate.so.1 (0xf7763000)
libcwbcore.so => /opt/ibm/iSeriesAccess/lib/libcwbcore.so (0xf7557000)
libodbcinst.so.1 => /usr/lib/i386-linux-gnu/libodbcinst.so.1 (0xf7544000)
libdl.so.2 => /lib/i386-linux-gnu/i686/cmov/libdl.so.2 (0xf753e000)
libpthread.so.0 => /lib/i386-linux-gnu/i686/cmov/libpthread.so.0 (0xf7522000)
libstdc++.so.6 => /usr/lib/i386-linux-gnu/libstdc++.so.6 (0xf7430000)
libm.so.6 => /lib/i386-linux-gnu/i686/cmov/libm.so.6 (0xf73ea000)
libgcc_s.so.1 => /lib/i386-linux-gnu/libgcc_s.so.1 (0xf73cd000)
libc.so.6 => /lib/i386-linux-gnu/i686/cmov/libc.so.6 (0xf721f000)
librt.so.1 => /lib/i386-linux-gnu/i686/cmov/librt.so.1 (0xf7216000)
libltdl.so.7 => /usr/lib/i386-linux-gnu/libltdl.so.7 (0xf720a000)
/lib/ld-linux.so.2 (0xf7766000)

正如你所看到的,没有遗漏libs。我错过了什么?

解决方案

最后,我安装了32位版本的Debian。在我看来,这个问题依赖于64位版本的IBM I Access Driver。我只是在新装置上遵循了同样的步骤,它就像一个魅力。希望它能帮助其他人。

这是#21286589的副本。没有NULL值并不意味着在使用不匹配的ABI时不会遇到问题。

这里的问题是,旧的驱动程序只设置64位指示符值的32位,PHP读取整个64位。在这里,PHP看到值140707423584261,它是十六进制的0x7FF900000005。您可以看到最后4个字节是0x00000005,它是5,应该是返回的实际数据的长度。其余部分是垃圾,因为驱动程序没有更改这些字节。

正如我在回答Linux odbc致命错误时提到的:允许的内存大小,您需要从IBM I Access Client Solutions Linux应用程序包中获得新的odbc驱动程序。此驱动程序遵循unixODBC自2.2.14以来一直使用的完整64位ABI。新的驱动程序包还包含Debian.deb包,所以您引用的博客中几乎所有的步骤都不再需要了。

相关内容

  • 没有找到相关文章

最新更新