/java-1.7.0-openjdk/openjdk/jdk/src/share/native/sun/security/smartcardio/pcsc.c

# · C · 381 lines · 294 code · 56 blank · 31 comment · 22 complexity · 2bad4ee399c5d2e422188d67ae48caf5 MD5 · raw file

  1. /*
  2. * Copyright (c) 2005, 2006, Oracle and/or its affiliates. All rights reserved.
  3. * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
  4. *
  5. * This code is free software; you can redistribute it and/or modify it
  6. * under the terms of the GNU General Public License version 2 only, as
  7. * published by the Free Software Foundation. Oracle designates this
  8. * particular file as subject to the "Classpath" exception as provided
  9. * by Oracle in the LICENSE file that accompanied this code.
  10. *
  11. * This code is distributed in the hope that it will be useful, but WITHOUT
  12. * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
  13. * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
  14. * version 2 for more details (a copy is included in the LICENSE file that
  15. * accompanied this code).
  16. *
  17. * You should have received a copy of the GNU General Public License version
  18. * 2 along with this work; if not, write to the Free Software Foundation,
  19. * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
  20. *
  21. * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
  22. * or visit www.oracle.com if you need additional information or have any
  23. * questions.
  24. */
  25. /* disable asserts in product mode */
  26. #ifndef DEBUG
  27. #ifndef NDEBUG
  28. #define NDEBUG
  29. #endif
  30. #endif
  31. #include <stdio.h>
  32. #include <stdlib.h>
  33. #include <string.h>
  34. #include <assert.h>
  35. #include <winscard.h>
  36. // #define J2PCSC_DEBUG
  37. #ifdef J2PCSC_DEBUG
  38. #define dprintf(s) printf(s)
  39. #define dprintf1(s, p1) printf(s, p1)
  40. #define dprintf2(s, p1, p2) printf(s, p1, p2)
  41. #define dprintf3(s, p1, p2, p3) printf(s, p1, p2, p3)
  42. #else
  43. #define dprintf(s)
  44. #define dprintf1(s, p1)
  45. #define dprintf2(s, p1, p2)
  46. #define dprintf3(s, p1, p2, p3)
  47. #endif
  48. #include "sun_security_smartcardio_PCSC.h"
  49. #include "pcsc_md.h"
  50. #define MAX_STACK_BUFFER_SIZE 8192
  51. // make the buffers larger than what should be necessary, just in case
  52. #define ATR_BUFFER_SIZE 128
  53. #define READERNAME_BUFFER_SIZE 128
  54. #define RECEIVE_BUFFER_SIZE MAX_STACK_BUFFER_SIZE
  55. #define J2PCSC_EXCEPTION_NAME "sun/security/smartcardio/PCSCException"
  56. void throwPCSCException(JNIEnv* env, LONG code) {
  57. jclass pcscClass;
  58. jmethodID constructor;
  59. jthrowable pcscException;
  60. pcscClass = (*env)->FindClass(env, J2PCSC_EXCEPTION_NAME);
  61. assert(pcscClass != NULL);
  62. constructor = (*env)->GetMethodID(env, pcscClass, "<init>", "(I)V");
  63. assert(constructor != NULL);
  64. pcscException = (jthrowable) (*env)->NewObject(env, pcscClass, constructor, (jint)code);
  65. (*env)->Throw(env, pcscException);
  66. }
  67. jboolean handleRV(JNIEnv* env, LONG code) {
  68. if (code == SCARD_S_SUCCESS) {
  69. return JNI_FALSE;
  70. } else {
  71. throwPCSCException(env, code);
  72. return JNI_TRUE;
  73. }
  74. }
  75. JNIEXPORT jint JNICALL JNI_OnLoad(JavaVM *vm, void *reserved) {
  76. return JNI_VERSION_1_4;
  77. }
  78. JNIEXPORT jlong JNICALL Java_sun_security_smartcardio_PCSC_SCardEstablishContext
  79. (JNIEnv *env, jclass thisClass, jint dwScope)
  80. {
  81. SCARDCONTEXT context;
  82. LONG rv;
  83. dprintf("-establishContext\n");
  84. rv = CALL_SCardEstablishContext(dwScope, NULL, NULL, &context);
  85. if (handleRV(env, rv)) {
  86. return 0;
  87. }
  88. // note: SCARDCONTEXT is typedef'd as long, so this works
  89. return (jlong)context;
  90. }
  91. /**
  92. * Convert a multi string to a java string array,
  93. */
  94. jobjectArray pcsc_multi2jstring(JNIEnv *env, char *spec) {
  95. jobjectArray result;
  96. jclass stringClass;
  97. char *cp, **tab;
  98. jstring js;
  99. int cnt = 0;
  100. cp = spec;
  101. while (*cp != 0) {
  102. cp += (strlen(cp) + 1);
  103. ++cnt;
  104. }
  105. tab = (char **)malloc(cnt * sizeof(char *));
  106. cnt = 0;
  107. cp = spec;
  108. while (*cp != 0) {
  109. tab[cnt++] = cp;
  110. cp += (strlen(cp) + 1);
  111. }
  112. stringClass = (*env)->FindClass(env, "java/lang/String");
  113. assert(stringClass != NULL);
  114. result = (*env)->NewObjectArray(env, cnt, stringClass, NULL);
  115. while (cnt-- > 0) {
  116. js = (*env)->NewStringUTF(env, tab[cnt]);
  117. (*env)->SetObjectArrayElement(env, result, cnt, js);
  118. }
  119. free(tab);
  120. return result;
  121. }
  122. JNIEXPORT jobjectArray JNICALL Java_sun_security_smartcardio_PCSC_SCardListReaders
  123. (JNIEnv *env, jclass thisClass, jlong jContext)
  124. {
  125. SCARDCONTEXT context = (SCARDCONTEXT)jContext;
  126. LONG rv;
  127. LPTSTR mszReaders;
  128. DWORD size;
  129. jobjectArray result;
  130. dprintf1("-context: %x\n", context);
  131. rv = CALL_SCardListReaders(context, NULL, NULL, &size);
  132. if (handleRV(env, rv)) {
  133. return NULL;
  134. }
  135. dprintf1("-size: %d\n", size);
  136. mszReaders = malloc(size);
  137. rv = CALL_SCardListReaders(context, NULL, mszReaders, &size);
  138. if (handleRV(env, rv)) {
  139. free(mszReaders);
  140. return NULL;
  141. }
  142. dprintf1("-String: %s\n", mszReaders);
  143. result = pcsc_multi2jstring(env, mszReaders);
  144. free(mszReaders);
  145. return result;
  146. }
  147. JNIEXPORT jlong JNICALL Java_sun_security_smartcardio_PCSC_SCardConnect
  148. (JNIEnv *env, jclass thisClass, jlong jContext, jstring jReaderName,
  149. jint jShareMode, jint jPreferredProtocols)
  150. {
  151. SCARDCONTEXT context = (SCARDCONTEXT)jContext;
  152. LONG rv;
  153. LPCTSTR readerName;
  154. SCARDHANDLE card;
  155. DWORD proto;
  156. readerName = (*env)->GetStringUTFChars(env, jReaderName, NULL);
  157. rv = CALL_SCardConnect(context, readerName, jShareMode, jPreferredProtocols, &card, &proto);
  158. (*env)->ReleaseStringUTFChars(env, jReaderName, readerName);
  159. dprintf1("-cardhandle: %x\n", card);
  160. dprintf1("-protocol: %d\n", proto);
  161. if (handleRV(env, rv)) {
  162. return 0;
  163. }
  164. return (jlong)card;
  165. }
  166. JNIEXPORT jbyteArray JNICALL Java_sun_security_smartcardio_PCSC_SCardTransmit
  167. (JNIEnv *env, jclass thisClass, jlong jCard, jint protocol,
  168. jbyteArray jBuf, jint jOfs, jint jLen)
  169. {
  170. SCARDHANDLE card = (SCARDHANDLE)jCard;
  171. LONG rv;
  172. SCARD_IO_REQUEST sendPci;
  173. unsigned char *sbuf;
  174. unsigned char rbuf[RECEIVE_BUFFER_SIZE];
  175. DWORD rlen = RECEIVE_BUFFER_SIZE;
  176. int ofs = (int)jOfs;
  177. int len = (int)jLen;
  178. jbyteArray jOut;
  179. sendPci.dwProtocol = protocol;
  180. sendPci.cbPciLength = sizeof(SCARD_IO_REQUEST);
  181. sbuf = (unsigned char *) ((*env)->GetByteArrayElements(env, jBuf, NULL));
  182. rv = CALL_SCardTransmit(card, &sendPci, sbuf + ofs, len, NULL, rbuf, &rlen);
  183. (*env)->ReleaseByteArrayElements(env, jBuf, (jbyte *)sbuf, JNI_ABORT);
  184. if (handleRV(env, rv)) {
  185. return NULL;
  186. }
  187. jOut = (*env)->NewByteArray(env, rlen);
  188. (*env)->SetByteArrayRegion(env, jOut, 0, rlen, (jbyte *)rbuf);
  189. return jOut;
  190. }
  191. JNIEXPORT jbyteArray JNICALL Java_sun_security_smartcardio_PCSC_SCardStatus
  192. (JNIEnv *env, jclass thisClass, jlong jCard, jbyteArray jStatus)
  193. {
  194. SCARDHANDLE card = (SCARDHANDLE)jCard;
  195. LONG rv;
  196. char readerName[READERNAME_BUFFER_SIZE];
  197. DWORD readerLen = READERNAME_BUFFER_SIZE;
  198. unsigned char atr[ATR_BUFFER_SIZE];
  199. DWORD atrLen = ATR_BUFFER_SIZE;
  200. DWORD state;
  201. DWORD protocol;
  202. jbyteArray jArray;
  203. jbyte tmp;
  204. rv = CALL_SCardStatus(card, readerName, &readerLen, &state, &protocol, atr, &atrLen);
  205. if (handleRV(env, rv)) {
  206. return NULL;
  207. }
  208. dprintf1("-reader: %s\n", readerName);
  209. dprintf1("-status: %d\n", state);
  210. dprintf1("-protocol: %d\n", protocol);
  211. jArray = (*env)->NewByteArray(env, atrLen);
  212. (*env)->SetByteArrayRegion(env, jArray, 0, atrLen, (jbyte *)atr);
  213. tmp = (jbyte)state;
  214. (*env)->SetByteArrayRegion(env, jStatus, 0, 1, &tmp);
  215. tmp = (jbyte)protocol;
  216. (*env)->SetByteArrayRegion(env, jStatus, 1, 1, &tmp);
  217. return jArray;
  218. }
  219. JNIEXPORT void JNICALL Java_sun_security_smartcardio_PCSC_SCardDisconnect
  220. (JNIEnv *env, jclass thisClass, jlong jCard, jint jDisposition)
  221. {
  222. SCARDHANDLE card = (SCARDHANDLE)jCard;
  223. LONG rv;
  224. rv = CALL_SCardDisconnect(card, jDisposition);
  225. dprintf1("-disconnect: 0x%X\n", rv);
  226. handleRV(env, rv);
  227. return;
  228. }
  229. JNIEXPORT jintArray JNICALL Java_sun_security_smartcardio_PCSC_SCardGetStatusChange
  230. (JNIEnv *env, jclass thisClass, jlong jContext, jlong jTimeout,
  231. jintArray jCurrentState, jobjectArray jReaderNames)
  232. {
  233. SCARDCONTEXT context = (SCARDCONTEXT)jContext;
  234. LONG rv;
  235. int readers = (*env)->GetArrayLength(env, jReaderNames);
  236. SCARD_READERSTATE *readerState = malloc(readers * sizeof(SCARD_READERSTATE));
  237. int i;
  238. jintArray jEventState;
  239. int *currentState = (*env)->GetIntArrayElements(env, jCurrentState, NULL);
  240. for (i = 0; i < readers; i++) {
  241. jobject jReaderName = (*env)->GetObjectArrayElement(env, jReaderNames, i);
  242. readerState[i].szReader = (*env)->GetStringUTFChars(env, jReaderName, NULL);
  243. readerState[i].pvUserData = NULL;
  244. readerState[i].dwCurrentState = currentState[i];
  245. readerState[i].dwEventState = SCARD_STATE_UNAWARE;
  246. readerState[i].cbAtr = 0;
  247. }
  248. (*env)->ReleaseIntArrayElements(env, jCurrentState, currentState, JNI_ABORT);
  249. rv = CALL_SCardGetStatusChange(context, (DWORD)jTimeout, readerState, readers);
  250. jEventState = (*env)->NewIntArray(env, readers);
  251. for (i = 0; i < readers; i++) {
  252. jint eventStateTmp;
  253. jobject jReaderName = (*env)->GetObjectArrayElement(env, jReaderNames, i);
  254. dprintf3("-reader status %s: 0x%X, 0x%X\n", readerState[i].szReader,
  255. readerState[i].dwCurrentState, readerState[i].dwEventState);
  256. (*env)->ReleaseStringUTFChars(env, jReaderName, readerState[i].szReader);
  257. eventStateTmp = (jint)readerState[i].dwEventState;
  258. (*env)->SetIntArrayRegion(env, jEventState, i, 1, &eventStateTmp);
  259. }
  260. free(readerState);
  261. handleRV(env, rv);
  262. return jEventState;
  263. }
  264. JNIEXPORT void JNICALL Java_sun_security_smartcardio_PCSC_SCardBeginTransaction
  265. (JNIEnv *env, jclass thisClass, jlong jCard)
  266. {
  267. SCARDHANDLE card = (SCARDHANDLE)jCard;
  268. LONG rv;
  269. rv = CALL_SCardBeginTransaction(card);
  270. dprintf1("-beginTransaction: 0x%X\n", rv);
  271. handleRV(env, rv);
  272. return;
  273. }
  274. JNIEXPORT void JNICALL Java_sun_security_smartcardio_PCSC_SCardEndTransaction
  275. (JNIEnv *env, jclass thisClass, jlong jCard, jint jDisposition)
  276. {
  277. SCARDHANDLE card = (SCARDHANDLE)jCard;
  278. LONG rv;
  279. rv = CALL_SCardEndTransaction(card, jDisposition);
  280. dprintf1("-endTransaction: 0x%X\n", rv);
  281. handleRV(env, rv);
  282. return;
  283. }
  284. JNIEXPORT jbyteArray JNICALL Java_sun_security_smartcardio_PCSC_SCardControl
  285. (JNIEnv *env, jclass thisClass, jlong jCard, jint jControlCode, jbyteArray jSendBuffer)
  286. {
  287. SCARDHANDLE card = (SCARDHANDLE)jCard;
  288. LONG rv;
  289. jbyte* sendBuffer = (*env)->GetByteArrayElements(env, jSendBuffer, NULL);
  290. jint sendBufferLength = (*env)->GetArrayLength(env, jSendBuffer);
  291. jbyte receiveBuffer[MAX_STACK_BUFFER_SIZE];
  292. jint receiveBufferLength = MAX_STACK_BUFFER_SIZE;
  293. ULONG returnedLength = 0;
  294. jbyteArray jReceiveBuffer;
  295. #ifdef J2PCSC_DEBUG
  296. {
  297. int k;
  298. printf("-control: 0x%X\n", jControlCode);
  299. printf("-send: ");
  300. for (k = 0; k < sendBufferLength; k++) {
  301. printf("%02x ", sendBuffer[k]);
  302. }
  303. printf("\n");
  304. }
  305. #endif
  306. rv = CALL_SCardControl(card, jControlCode, sendBuffer, sendBufferLength,
  307. receiveBuffer, receiveBufferLength, &returnedLength);
  308. (*env)->ReleaseByteArrayElements(env, jSendBuffer, sendBuffer, JNI_ABORT);
  309. if (handleRV(env, rv)) {
  310. return NULL;
  311. }
  312. #ifdef J2PCSC_DEBUG
  313. {
  314. int k;
  315. printf("-recv: ");
  316. for (k = 0; k < returnedLength; k++) {
  317. printf("%02x ", receiveBuffer[k]);
  318. }
  319. printf("\n");
  320. }
  321. #endif
  322. jReceiveBuffer = (*env)->NewByteArray(env, returnedLength);
  323. (*env)->SetByteArrayRegion(env, jReceiveBuffer, 0, returnedLength, receiveBuffer);
  324. return jReceiveBuffer;
  325. }