我正在尝试使用带有以下代码的javax.smartcardio API加载智能卡终端:
public CardTerminal getReadyCardTerminal() throws CardException {
TerminalFactory factory = TerminalFactory.getDefault();
CardTerminals terminals = factory.terminals();
List<CardTerminal> list = terminals.list(State.CARD_PRESENT);
while (list.isEmpty()) {
terminals.waitForChange(1000);
list = terminals.list(State.CARD_PRESENT);
}
CardTerminal cardTerminal = list.get(0);
return cardTerminal;
}
...我总是得到以下异常:
java.lang.IllegalStateException: no terminals
at javax.smartcardio.TerminalFactory$NoneCardTerminals.waitForChange(TerminalFactory.java:145)
在Windows Vista / 7上,一切正常,但我无法在Linux上运行。我正在使用Ubuntu 12.04 64位。
我使用以下命令安装了pcscd服务:
sudo apt-get install libccid pcscd libpcsclite-dev libpcsclite1
sudo service pcscd start
pcsc_scan命令打印出来:
PC/SC device scanner
V 1.4.18 (c) 2001-2011, Ludovic Rousseau <ludovic.rousseau@free.fr>
Compiled with PC/SC lite version: 1.7.4
Using reader plug'n play mechanism
Scanning present readers...
0: OMNIKEY CardMan 3x21 00 00
Tue Sep 11 15:44:49 2012
Reader 0: OMNIKEY CardMan 3x21 00 00
Card state: Card inserted,
ATR: <some hexa codes>
...
所以一切看起来都不错,但智能卡只是不起作用。我正在尝试使用Oracle和OpenJDK 1.7.0_05,32和64位。
在Ubuntu 32位环境下,代码与OpenJDK(但不是Oracle JDK,不知道为什么)运行正常。所以我认为从Java到PC / SC库的64位桥接器存在问题。
有什么想法吗?
感谢。
答案 0 :(得分:35)
我想我找到了一个解决方法,因为我遇到了类似的问题。在bugreport from ubuntu中,它表示javax.smartcardio库在错误的目录中搜索PC / SC库。
通过在我的机器上指定PC / SC库的路径,就像bugreport提到的那样,我得到了它。
bug报告中的路径对我来说是错误的,我在64位fedora上,其中pc / sc库安装在/usr/lib64/libpcsclite.so.1
所以我的解决方法是指定java的库路径,如下所示:
java -Dsun.security.smartcardio.library=/usr/lib64/libpcsclite.so.1
根据您的Linux发行版,libpcsclite.so.1
的位置实际上可能不同,它也可能位于/lib/x86_64-linux-gnu/libpcsclite.so.1
(即Kubuntu 15.04)。
在这种情况下,请这样称呼:
java -Dsun.security.smartcardio.library=/lib/x86_64-linux-gnu/libpcsclite.so.1
答案 1 :(得分:7)
我正在使用带有debian arm版本的覆盆子
首先找到libpcsclite的位置:
$ ldd -r /usr/bin/pcsc_scan
然后使用libpcsclite位置:
java -Dsun.security.smartcardio.library=/usr/lib/arm-linux-gnueabihf/libpcsclite.so.1
答案 2 :(得分:4)
对于其他在64位计算机上使用Ubuntu 14进行此操作的人。我发现.so文件实际上位于以下目录中
/usr/lib/x86_64-linux-gnu/libpcsclite.so
因此,使用以下设置运行我的应用程序对我有用
-Dsun.security.smartcardio.library = / usr / lib中/ x86_64的-Linux的GNU / libpcsclite.so
答案 3 :(得分:3)
您需要在调用程序时给出libpcsclite.so.1的路径,如下所示
java -Dsun.security.smartcardio.library=/path/to/libpcsclite.so.1
如果您不知道库的路径,请使用以下命令
find /usr/lib -name libpcsclite.so.1
这通常会显示您计算机上的路径。我在Ubuntu 10(32位)和Ubuntu 15(32位和64位)上使用它
如果你像我一样懒,那么你可以做的就是在使用javax.smartcardio库之前在程序中包含这部分代码
try {
String comm[] = { "find", "/usr", "/lib", "-name",
"libpcsclite.so.1" };
Process p = Runtime.getRuntime().exec(comm);
BufferedReader reader = new BufferedReader(
new InputStreamReader(p.getInputStream()));
while ((line = reader.readLine()) != null && !line.equals("")) {
if (line.contains("libpcsclite.so.1")) {
System.setProperty("sun.security.smartcardio.library",line);
break;
}
}
p.waitFor();
} catch (Exception e) {
e.printStackTrace();
}
现在,您可以像往常一样运行代码,而不包括libpcsclite.so.1
的路径答案 4 :(得分:1)
添加解决方案,提供路径作为参数,如下所示:
java -Dsun.security.smartcardio.library=/usr/lib64/libpcsclite.so.1
如果您不想在每次调用JVM时都提供此项,请在环境变量_JAVA_OPTIONS和/或JAVA_OPTS中设置它:
export _JAVA_OPTIONS="-Dsun.security.smartcardio.library=/usr/lib64/libpcsclite.so.1"
export JAVA_OPTS="-Dsun.security.smartcardio.library=/usr/lib64/libpcsclite.so.1"
由于这是影响整个系统的bug的解决方法,因此恕我直言也可以在全系统范围内应用此解决方法。
JAVA_OPTS具有本地范围,必须由运行代码的脚本进行评估; _JAVA_OPTIONS应该由JRE自动评估。
答案 5 :(得分:1)
另一种方法(我最喜欢的)是制作一些符号链接。
它的优点是它可以在系统范围内工作(没有jvm参数,没有环境变量)。
对于我(心爱的)debian jessie amd64:
ln -s /usr/lib/x86_64-linux-gnu/libpcsclite.so libpcsclite.so
ln -s /usr/lib/x86_64-linux-gnu/libpcsclite.so.1 libpcsclite.so.1
ln -s /usr/lib/x86_64-linux-gnu/libpcsclite.so.1.0.0 libpcsclite.so.1.0.0
注意:这可能需要超级用户访问。
答案 6 :(得分:0)
补充@AshanPerera的答案,因为有时每次搜索可能很慢,因此您可以在第一次搜索时将它们存储在文件中,然后从头开始读取它:
try {
String filename = "libpcsclite.location";
File propertyFile = new File(filename);
if(propertyFile.createNewFile())
{
String commandWithArguments[] = { "find", "/usr", "/lib", "-name","libpcsclite.so.1" };
Process searchProcess = Runtime.getRuntime().exec(commandWithArguments);
BufferedReader searchReader = new BufferedReader(new InputStreamReader(searchProcess.getInputStream()));
String propertyValue;
while ( (propertyValue = searchReader.readLine()) != null && !propertyValue.equals(""))
{
if (propertyValue.contains("libpcsclite.so.1")) {
BufferedWriter propertyWriter = new BufferedWriter(new OutputStreamWriter(new FileOutputStream(propertyFile)));
propertyWriter.write(propertyValue);
propertyWriter.close();
System.setProperty("sun.security.smartcardio.library",propertyValue);
break;
}
}
searchProcess.waitFor();
}
else
{
BufferedReader propertyReader = new BufferedReader(new InputStreamReader(new FileInputStream(propertyFile)));
String propertyValue = propertyReader.readLine();
System.setProperty("sun.security.smartcardio.library",propertyValue);
}
} catch (Exception e) {
e.printStackTrace();
}