Notes on using a CAC/PIV for PopOS!
pcsc-tools
libccid
libpcsc-perl
libpcsclite1
pcscd
vsmartcard-vpcd
libnss3-tools
opensc
opensc-pkcs11
Some important setup information. Most guides will direct your towards CoolKey or CACKey as the pkcs11 management software to interact with your CAC. CoolKey used to be very effective but it cannot interact with the PIV certificates on your CAC; CACKey has the distinguishment of only being available to download after already being authenticated via CAC… also it’s just not very good. This leaves us with OpenSC. OpenSC is the bee’s knees, it can interact with PIV certificates, is very actively updated, RedHat provides upstream commits, and is very fast! CACKey and CoolKey must be completely uninstalled.
The latest version of OpenSC on the Debian repositories is v0.21. I personally have had trouble with this version so I built and installed v0.22 from source from their github page. You can follow those instructions at the following link: https://github.com/OpenSC/OpenSC/wiki/Compiling-and-Installing-on-Unix-flavors
The following command will get you up and running as far as dependencies go:
sudo apt remove cackey coolkey libckyapplet1 libckyapplet1-dev -y && sudo apt purge -y
sudo apt update -y && sudo apt upgrade -y && sudo apt install pcsc-tools libccid libpcsc-perl libpcsclite1 pcscd opensc opensc-pkcs11 vsmartcard-vpcd libnss3-tools -y
Now that all your dependencies are installed it’s time to start checking if things are operating how they should. The first step is to ensure your middleware actually sees your CAC; this is done via pcscd and it’s associated tools.
Running pcsc_scan should give the following output:
pcsc_scan
Using reader plug n play mechanism
Scanning present readers...
0: Virtual PCD 00 00
1: Virtual PCD 00 01
2: Alcor Micro AU9540 00 00
3: Yubico YubiKey OTP+FIDO+CCID 01 00
Tue Sep 28 10:10:57 2021
Reader 0: Virtual PCD 00 00
Event number: 0
Card state: Card removed,
Reader 1: Virtual PCD 00 01
Event number: 0
Card state: Card removed,
Reader 2: Alcor Micro AU9540 00 00
Event number: 7
Card state: Card inserted, Shared Mode,
ATR: 3B DB 96 00 80 1F 03 00 31 C0 64 B0 F3 10 00 07 90 00 80
ATR: 3B DB 96 00 80 1F 03 00 31 C0 64 B0 F3 10 00 07 90 00 80
+ TS = 3B --> Direct Convention
+ T0 = DB, Y(1): 1101, K: 11 (historical bytes)
TA(1) = 96 --> Fi=512, Di=32, 16 cycles/ETU
250000 bits/s at 4 MHz, fMax for Fi = 5 MHz => 312500 bits/s
TC(1) = 00 --> Extra guard time: 0
TD(1) = 80 --> Y(i+1) = 1000, Protocol T = 0
-----
TD(2) = 1F --> Y(i+1) = 0001, Protocol T = 15 - Global interface bytes following
-----
TA(3) = 03 --> Clock stop: not supported - Class accepted by the card: (3G) A 5V B 3V
+ Historical bytes: 00 31 C0 64 B0 F3 10 00 07 90 00
Category indicator byte: 00 (compact TLV data object)
Tag: 3, len: 1 (card service data byte)
Card service data byte: C0
- Application selection: by full DF name
- Application selection: by partial DF name
- EF.DIR and EF.ATR access services: by GET RECORD(s) command
- Card with MF
Tag: 6, len: 4 (pre-issuing data)
Data: B0 F3 10 00
Mandatory status indicator (3 last bytes)
LCS (life card cycle): 07 (Operational state (activated))
SW: 9000 (Normal processing.)
+ TCK = 80 (correct checksum)
Possibly identified card (using /usr/share/pcsc/smartcard_list.txt):
3B DB 96 00 80 1F 03 00 31 C0 64 B0 F3 10 00 07 90 00 80
DoD CAC, Oberthur ID One 128 v5.5 Dual
If you get different output first check if the daemon is actually running:
sudo systemctl status pcscd.s*
● pcscd.service - PC/SC Smart Card Daemon
Loaded: loaded (/lib/systemd/system/pcscd.service; indirect; vendor preset: enabled)
Active: active (running) since Sun 2021-09-26 20:37:17 EDT; 1 day 13h ago
TriggeredBy: ● pcscd.socket
Docs: man:pcscd(8)
Main PID: 2022 (pcscd)
Tasks: 11 (limit: 37695)
Memory: 2.6M
CGroup: /system.slice/pcscd.service
└─2022 /usr/sbin/pcscd --foreground --auto-exit
Sep 26 21:43:14 pop-os pcscd[2022]: 12411648 ifdwrapper.c:477:IFDControl() Card not transacted: 606
Sep 26 21:43:14 pop-os pcscd[2022]: 00000112 ifdwrapper.c:477:IFDControl() Card not transacted: 606
Sep 26 22:02:58 pop-os pcscd[2022]: 99999999 ifdwrapper.c:477:IFDControl() Card not transacted: 606
Sep 26 22:02:58 pop-os pcscd[2022]: 00000065 ifdwrapper.c:477:IFDControl() Card not transacted: 606
Sep 26 22:09:01 pop-os pcscd[2022]: 99999999 ifdwrapper.c:477:IFDControl() Card not transacted: 606
Sep 26 22:09:01 pop-os pcscd[2022]: 00000181 ifdwrapper.c:477:IFDControl() Card not transacted: 606
Sep 26 22:55:15 pop-os pcscd[2022]: 99999999 ifdwrapper.c:477:IFDControl() Card not transacted: 606
Sep 26 22:55:15 pop-os pcscd[2022]: 00000256 ifdwrapper.c:477:IFDControl() Card not transacted: 606
Sep 27 11:07:28 pop-os pcscd[2022]: 99999999 ifdwrapper.c:477:IFDControl() Card not transacted: 606
Sep 27 11:07:28 pop-os pcscd[2022]: 00000068 ifdwrapper.c:477:IFDControl() Card not transacted: 606
● pcscd.socket - PC/SC Smart Card Daemon Activation Socket
Loaded: loaded (/lib/systemd/system/pcscd.socket; enabled; vendor preset: enabled)
Active: active (running) since Sun 2021-09-26 20:37:12 EDT; 1 day 13h ago
Triggers: ● pcscd.service
Listen: /run/pcscd/pcscd.comm (Stream)
Tasks: 0 (limit: 37695)
Memory: 0B
CGroup: /system.slice/pcscd.socket
Sep 26 20:37:12 pop-os systemd[1]: Listening on PC/SC Smart Card Daemon Activation Socket.
If there are errors restart the daemon:
sudo systemctl restart pcscd.socket && sudo systemctl restart pcscd.service
The last trouble shooting tip is to execute the following command; this will unload the kernel modules and allow whatever is plugged into the usb slot to claim the usb slot. This is generally only useful if you get the following message from the pcsc_scan
: "scanning present readers waiting for the first reader..."
modprobe -r pn533 nfc
Now that the middleware is operational we can pivot to unlocking and interacting with our CAC. First use opensc-tools
to query some information about your CAC; the following command will list driver information available to OpenSC and if OpenSC can interact with the middleware. This is important and probably the largest stumbling block!:
opensc-tool -l
# Detected readers (pcsc)
Nr. Card Features Name
0 No Virtual PCD 00 00
1 No Virtual PCD 00 01
2 Yes Alcor Micro AU9540 00 00
3 Yes Yubico YubiKey OTP+FIDO+CCID 01 00
---------------------------------------------------------
opensc-tool -D
Available card drivers:
cac Common Access Card (CAC)
If your CAC reader is not listed under Detected readers (pcsc)
there is a problem between OpenSC and pcscd. I do not know of any trouble shooting steps for a problem here except reading the source and reinstalling.
If you do not see the cac drivers under Available card drivers
then we need to force OpenSC to use them. I've found that OpenSC on PopOS generally has this problem but thankfully it's super easy to fix.
Edit /etc/opensc/opensc.conf
and add the following lines:
card_drivers = cac
force_card_driver = cac
So your opensc.conf
should look like this:
cat opensc.conf
app default {
# debug = 3;
# debug_file = opensc-debug.txt;
card_drivers = cac
force_card_driver = cac
framework pkcs15 {
# use_file_caching = true;
}
}
Query your CAC with opensc-tools again and see if it detects the drivers.
Now we can start working on the certificates and preparing the browers to use your CAC.
I will show instructions on how to prepare both FireFox and Microsoft Edge (Chromium) on PopOS!
Download the DoD certificates and unzip them:
https://public.cyber.mil/pki-pke/pkipke-document-library/
This command will obviously have to be updated as Cyber.mil continuously releases new bundles and moves them around on their website:
wget https://dl.dod.cyber.mil/wp-content/uploads/pki-pke/zip/certificates_pkcs7_DoD.zip && unzip certificates_pkcs7_DoD.zip
Load the proper security device on FireFox:
There are two ways to register the security device with Firefox, a manual and automatic way. I recommend the automatic way:
pkcs11-register
Or the manual way: Navigate to Edit -> Preference -> Advanced -> Certificates -> Security Devices. You should see the following entry under modules and devices:
If you installed your OpenSC from source then the path to your onepin-opensc-pkcs11.so
will be exactly as mine. If you installed OpenSC from the PopOS! package manager the path will be /usr/lib/x86_64-linux-gnu/pkcs11/onepin-opensc-pkcs11.so
If you need to manually add the security device do the following:
Navigate to Edit -> Preference -> Advanced -> Certificates -> Security Devices.
You should see the following entry under modules and devices: and click "Load" to load a module using
`/usr/lib/x86_64-linux-gnu/pkcs11/onepin-opensc-pkcs11.so` or `/usr/lib/onepin-opensc-pkcs11.so`.
Install the certificates from the mentioned zip-file in this order, by going to Edit -> Preference -> Advanced -> Certificates -> View Certificates -> Authorities -> Import (make sure to at-least check the box for "Trust this CA to identify websites"):
Note: As of the 5.7 version of the certificate zip
-
Certificates_PKCS7_v5.7_DoD.der.p7b
-
Certificates_PKCS7_v5.7_DoD_DoD_Root_CA_2.der.p7b
-
Certificates_PKCS7_v5.7_DoD_DoD_Root_CA_3.der.p7b
-
Certificates_PKCS7_v5.7_DoD_DoD_Root_CA_4.der.p7b
-
Certificates_PKCS7_v5.7_DoD_DoD_Root_CA_5.der.p7b
-
Certificates_PKCS7_v5.7_DoD.pem.p7b
Firefox is now ready to use.
Query nssdb to see if the OpenSC framework was registered in the NSS DB by using pkcs11-register
modutil -dbdir sql:$HOME/.pki/nssdb/ -list
You should see an entry similar to the following:
Listing of PKCS #11 Modules
-----------------------------------------------------------
1. NSS Internal PKCS #11 Module
uri: pkcs11:library-manufacturer=Mozilla%20Foundation;library-description=NSS%20Internal%20Crypto%20Services;library-version=3.61
slots: 2 slots attached
status: loaded
2. OpenSC smartcard framework (0.22)
library name: /usr/lib/onepin-opensc-pkcs11.so
uri: pkcs11:library-manufacturer=OpenSC%20Project;library-description=OpenSC%20smartcard%20framework;library-version=0.22
If OpenSC is not registered in NSS DB then you need to manually add it with the following command. It is very important you close all browsers before registering with the database. Change the 'x' to match your version number and the libfile to match your onepin-opensc-pkcs11.so
installation directory:
modutil -dbdir sql:$HOME/.pki/nssdb/ -add "OpenSC smartcard framework (0.2x)" -libfile /usr/lib/onepin-opensc-pkcs11.so
Query the database again and see if it's added.
You must not have any other residual nssdb entries from cackey, coolkey, or old versions of opensc. NSS Internal and OpenSC Framework should be the only entries. Remove other entries with the following command:
modutil -dbdir sql:$HOME/.pki/nssdb/ -delete "Name of Module (Probably CAC Module since everyone follows the milcac tutorial)"
Now you must add the DoD Certificates to nssdb. Navigate to the location of the unziped DoD Certificates and install via the following command:
for n in *.p7b; do certutil -d sql:$HOME/.pki/nssdb -A -t TC -n $n -i $n; done
for n in *.pem; do certutil -d sql:$HOME/.pki/nssdb -A -t TC -n $n -i $n; done
Or the manual way:
Re-open Chrome, Navigate to Settings -> Show Advanced Settings
-> Manage Certificates -> Authorities to load CA bundle from the PEM-formatted file from above.
Verify the authority is in Chrome under Settings -> Show Advanced Settings -> Manage Certificates -> Authorities then expand "org-U.S. Government" and you should see a number of "DoD" certificates listed.
Go to https://www.vmware.com/go/viewclients and download the latest 64 bit bundle
Make the installer executable and execute it with sudo permissions:
chmod +x VMware-Horizon-Client-2209-8.7.0-20616018.x64.bundle
sudo ./VMware-Horizon-Client-2209-8.7.0-20616018.x64.bundle
Create a symlink to opensc-pkcs11.so
sudo ln -s /usr/lib/x86_64-linux-gnu/opensc-pkcs11.so /usr/lib/vmware/view/pkcs11/libopenscpkcs11.so
Go to https://https://customerconnect.omnissa.com/downloads/#all_products and download and install the latest 64 bit horizon client for your Linux distribution
This process is documented here: https://docs.omnissa.com/bundle/HorizonClientLinuxGuideVmulti/page/ConfigureHorizonClientforSmartCardAuthentication.html The path needs to be slightly modified to accomodate the steps taken above.
# Create PKCS11 directory
sudo mkdir /usr/lib/omnissa/horizon/pkcs11
# Create symlink to opensc-pkcs11.so
sudo ln -s /usr/lib/x86_64-linux-gnu/opensc-pkcs11.so /usr/lib/omnissa/horizon/pkcs11/libopenscpkcs11.so
To not recieve an untrusted error when connecting to the horizon server, we need to add the dod certificates to the system certificate store.
Move to the directory you extracted the DoD certs to.
cd Certificates_PKCS7_v5.9_DoD
Use openssl to convert the pkcs bundle to a single pem file
openssl pkcs7 -print_certs -in Certificates_PKCS7_v5.9_DoD.pem.p7b -out dod_bundle.pem
Use awk to split the pem file into individual crt files
awk '
split_after == 1 {n++;split_after=0}
/-----END CERTIFICATE-----/ {split_after=1}
{print > "cert" n ".crt"}' < dod_bundle.pem
Create a dod subfolder in /usr/local/share/ca-certificates and copy the certificate files to it
sudo mkdir -p /usr/local/share/ca-certificates/dod/
sudo cp *.crt /usr/local/share/ca-certificates/dod/
Last, update the certificate store
sudo update-ca-certificates
You are now fully CAC enabled on Linux; I find Linux to be a more stable environment for CAC than Windows!