Smartcard terminal removal : SCARD_E_NO_SERVICE CardException

前端 未结 3 1763
北恋
北恋 2020-12-05 16:11

I am working on an Java application which uses smartcardio to work with smartcard. It must be possible to have one removing its USB card reader and then inserti

相关标签:
3条回答
  • 2020-12-05 16:47

    This post is quite old, but it has been useful for me to fix the problem described on Windows 8.

    The solution from JR Utily did not work fully: in case of a reader unplugged then plugged again, there were errors on the CardTerminal instance.

    So I added some code to clear the terminals list, as you can see in the code below.

            Class pcscterminal = Class.forName("sun.security.smartcardio.PCSCTerminals");
            Field contextId = pcscterminal.getDeclaredField("contextId");
            contextId.setAccessible(true);
    
            if(contextId.getLong(pcscterminal) != 0L)
            {
                // First get a new context value
                Class pcsc = Class.forName("sun.security.smartcardio.PCSC");
                Method SCardEstablishContext = pcsc.getDeclaredMethod(
                                                   "SCardEstablishContext",
                                                   new Class[] {Integer.TYPE }
                                               );
                SCardEstablishContext.setAccessible(true);
    
                Field SCARD_SCOPE_USER = pcsc.getDeclaredField("SCARD_SCOPE_USER");
                SCARD_SCOPE_USER.setAccessible(true);
    
                long newId = ((Long)SCardEstablishContext.invoke(pcsc, 
                        new Object[] { SCARD_SCOPE_USER.getInt(pcsc) }
                ));
                contextId.setLong(pcscterminal, newId);
    
    
                // Then clear the terminals in cache
                TerminalFactory factory = TerminalFactory.getDefault();
                CardTerminals terminals = factory.terminals();
                Field fieldTerminals = pcscterminal.getDeclaredField("terminals");
                fieldTerminals.setAccessible(true);
                Class classMap = Class.forName("java.util.Map");
                Method clearMap = classMap.getDeclaredMethod("clear");
    
                clearMap.invoke(fieldTerminals.get(terminals));
            }
    
    0 讨论(0)
  • 2020-12-05 16:49

    (This is just a comment but I don't have enough rep to post comments.)

    The service it refers to is the Windows Smart Card service, also known as the smart card resource manager. If you open the Services MMC console you'll see it there with the startup type set to Manual (Trigger Start). In Windows 8 this service was changed to run only while a smart card reader is attached to the system (to save resources) and the service is automatically stopped when the last reader is removed. Stopping the service invalidates any outstanding handles.

    The native Windows solution is to call SCardAccessStartedEvent and use the handle it returns to wait for the service to start before using SCardEstablishContext to connect to the resource manager again.

    0 讨论(0)
  • 2020-12-05 16:50

    I have found a way, but it's using reflective code. I'd rather found a cleaner method, but it seem there is no official API to manage Smart Card Context. All classes are private.

    The initContext() method of sun.security.smartcardio.PCSCTerminals (http://www.docjar.com/html/api/sun/security/smartcardio/PCSCTerminals.java.html) prevents new threads from getting a new context after the first one was initialized : the method is called, but the context is seen as a singleton and isn't re-initialzed.

    Passing through the private in everything around this with java.lang.reflect, it is possible to force the creation of a new context and store its new id as the "official" contextId. This should be done before instanciating the new TerminalFactory.

        // ...
        Class pcscterminal = Class.forName("sun.security.smartcardio.PCSCTerminals");
        Field contextId = pcscterminal.getDeclaredField("contextId");
        contextId.setAccessible(true);
    
        if(contextId.getLong(pcscterminal) != 0L)
        {
            Class pcsc = Class.forName("sun.security.smartcardio.PCSC");
            Method SCardEstablishContext = pcsc.getDeclaredMethod(
                                               "SCardEstablishContext",
                                               new Class[] {Integer.TYPE }
                                           );
            SCardEstablishContext.setAccessible(true);
    
            Field SCARD_SCOPE_USER = pcsc.getDeclaredField("SCARD_SCOPE_USER");
            SCARD_SCOPE_USER.setAccessible(true);
    
            long newId = ((Long)SCardEstablishContext.invoke(pcsc, 
                  new Object[] { Integer.valueOf(SCARD_SCOPE_USER.getInt(pcsc)) }
                  )).longValue();
            contextId.setLong(pcscterminal, newId);
        }
        // ...
    
    0 讨论(0)
提交回复
热议问题