我有一个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包,所以您引用的博客中几乎所有的步骤都不再需要了。