API
An API is an interface used at the source code level. In C langage, you include a header file and link to the associated library. In the PC/SC case you use something like:#ifdef __APPLE__ #include <PCSC/winscard.h> #include <PCSC/wintypes.h> #else #include <winscard.h> #endif
And then use the PC/SC functions as provided in the header file.
ABI
The ABI is an interface used at the binary level.At the API level the type
int
is used. At the ABI level the representation of int
is used. The difference is that the C language do not define what int
is. At the ABI level the representation of int
is fixed by the compiler.Evolution of API and/or ABI
The API may evolve if a new function is added, an existing function is removed or a function signature is changed (a function parameter is added or removed for example).Each time the API change the ABI also changes. The ABI may also evolve even if the API do not change. This is more rare but happened with C++ when GCC changed the way to pass parameters to a function/method.
To avoid incompatibility problems on GNU/Linux the library contains an versioning. It is called soname. Applications using the old API/ABI will use libfoo version n. Applications using the new API/ABI will use libfoo version n+1. It is possible to have the two library versions installed at the same time (I don't know if it is possible to do that on a Windows system).
The JVM problems
SUN/Oracle made 2 mistakes in its use of the PC/SC library.Direct use of libpcsclite.so
As written above a library is versioned. In the case of PC/SC the library is calledlibpcsclite.so.1
on a GNU/Linux system. The previous version was libpcsclite.so.0.
I changed the API in version 1.2.9-beta1 (May 2004). I then increased the ABI version from 0 to 1.The file
libpcsclite.so
is a symbolic link pointing to the version corresponding to the installed header files. Only the linker should use that file when building an application. On a Debian (or Ubuntu) system the file libpcsclite.so
is provided by the libpcsclite-dev
package and not by the libpcsclite1
package.The Oracle JVM tries to loads
libpcsclite.so
directly. This is wrong because:- This file is not installed by default when the PC/SC library is installed
- This file do not reference a particular library version. So if the PC/SC API change again then the JVM will miserably fail.
I get many bug reports because of that. But the problem is not on the pcsc-lite side. So I can't do much.
Direct definition of DWORD
Oracle think a
DWORD
is a 64-bit entity on a 64-bit system. This is not always the case and is wrong on Mac OS X in 64-bits mode.Apple defines
DWORD
as:typedef uint32_t DWORD;
On Linux it is defined as:
typedef unsigned long DWORD;
If the application (the JVM in this case) and the library (PCSC framework) do not agree on the ABI (the size of a
DWORD
parameter) then bad things happen.See the bug report javax.smartcardio package not working in Java 7 on OS X ML for a solution to this problem.
Conclusion
I wrote this blog article so that I can refer people at it. And so that you can refer Oracle at it.I do not use Java. I do not know how to report a JVM bug at Oracle. If you know then please send them a link of this article.