Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Provide OS integration to sign /boot and/or root during updates #1615

Open
JonathonHall-Purism opened this issue Feb 20, 2024 · 5 comments
Open

Comments

@JonathonHall-Purism
Copy link
Collaborator

A common paint point in Heads is that when any changes are made, it is difficult to tell if the changes are intentional or due to tampering.

In my opinion, the best solution to this is to eliminate the question. When intentional changes are made, sign them then as part of the update process.

It might be possible for Heads to include or offer integrations for common OSes. E.g. we could build packages for Debian-based, Fedora-based, etc. distributions that would know how to re-sign /boot after an update occurs. Potentially, this could be extended to root also for the root-hashing feature.

For Debian, it is possible to install apt/dpkg hooks that occur before or after dpkg invocations, package installation, etc. (one example: https://unix.stackexchange.com/questions/76294/hook-a-script-to-apt-get). Assuming other package managers offer similar functionality, we could probably export a Heads script and hook it at the appropriate place to integrate signing into the update process.

Alternative to #1599

@tlaurion
Copy link
Collaborator

tlaurion commented May 20, 2024

@DemiMarie this is the only real solution to the problem until unified signed kernels that kexec could verify signature against.

How would those hook script look like for qubesos dom0? Ideas on how to get there practically would be beneficial from the OS side.

More desirable alternative to #1599 and #1620

@DemiMarie
Copy link

@tlaurion what makes this more desirable than #1620?

One approach might be:

  1. Qubes OS provides a directory containing scripts that are run during each kernel update. The scripts are passed the kernel and initramfs as arguments.
  2. A script provided by Heads signs the kernel & initramfs using a key in the TPM. The key has policies that make it useable only when a verified kernel + initramfs combo has been booted.
  3. Either the signature or the code being signed contains a version number.
  4. The version number is compared against a TPM NV index, preventing downgrade attacks.

@tlaurion
Copy link
Collaborator

tlaurion commented May 24, 2024

@tlaurion what makes this more desirable than #1620?

@DemiMarie #1615 as opposed to #1620 would

  • have dom0 attest (sign) with user keys (as opposed to upstream being able to sign UKI as of now) something that could be validated by Heads with same public key matching user owned and protected private key never escaping usb security dongle and usb security dongle rate limiting and locking usb security dongle if 3 wrong user pin attempted to sign /boot components;
  • The same suggested process per Provide OS integration to sign /boot and/or root during updates #1615 (comment) would be applicable as hooks and generalizable to all OSes instead of being utopist about a more complicated approach which won't inform end users about tampered binaries being yes/no approach which doesn't help what the current proposition tries to accomplish (what changed, since when, what to investigate for)
  • Consider including git for /boot and root fs integrity and changes reporting #1599 (comment) was considered "untrusted" even if we would open rootfs as RO and use OS implied trusted integrity checks (yum/dnf integrity checks of files which should be trusted and verified: aka rpm -V) from their own integrity checks mechanisms.

I believe this (#1615) is the most practical solution of all proposed up to now in the goal of providing additional auditability, not another yes/no answer which doesn't help end-users and is not the goal of Heads, which goal is to extend auditability, not reduce nor opacify even more the security premises.


@DemiMarie In the goal of challenging your comment at #1599 (comment), again the ideal here would then be #1568 in the goal of #523 (firmware integrity related0 but this is not what we try to attain here: What we try to attain here is having dom0 (or any linux or even BSD if we dream enough) to be able to use hook scripts, deployed per OS internal requirements (OS installable packages) to effectively sign boot related material with user controlled keys, where Heads responsibility would be to validate (somehow) those signatures in a trustable fashion from a system that signed it before it became compromised.

I'm really not convinced here that requiring another nvram region is the answer to that need, where the OS should be responsible to enforce the signing of its boot components and where the bootloader responsibility should be to be to verify them somehow. That's the role Heads should take here, one way or the other, without the OS lacking signing boot components mechanisms being shoveled downstream (Heads), which should only be responsible to validate those, not implement those.

I hope we have a common understanding of the goals first without attempting to recreate the wheel, staying practical. I also hope that you have used Heads at least once with DEBUG mode activated, if not recurrently on a testing laptop under your control to understand that utopist ideas that cannot be implemented on Heads deployments (TPMv1+TPMv2, sealing not quotes) can only be be iteratively implemented with backward compatibility support. But Heads cannot resolve all ecosystem problems, while keeping its mission goal of having end users staying in control (owners of their machines, keys, secrets) and Heads providing oversight (Advanced oversight possible, see #1307) over what OS can do wrong between the moment it was healthy, and telling it is not, anymore.

Heads doesn't pretend being able to do what it can't do, even less in a scenario where security premises of the OS being launched doesn't enforce those mechanisms themselves somehow. This issue is about that: agreeing on a path to go, and then implement it in a way that fits the requirements. In this case for dom0, that would imply passing a usb security dongle to dom0 or sending binaries to a dvm or producing a hash of those binaries, have that hash file sent to dvm, that dvm be able to talk to the usb security dongle to detach sign those /boot digests and have place that detached signed digest back into /boot. Heads wouldn't care about who and how that hash file was detached signed, it would only care about the integrity+authentication that that detached signed digest provides, proving that it was done in collaboration by the end user and the fact that the end user was there (physical presence), that the usb security dongle was also present and that the user typed his private key passphrase to be able to detach sign that boot digest. This is what this issue is about for what is revlevant to QubesOS if this strategy is considered valid and actionnable, where QubesOS is the most complicated use case because of its security guarantees, and where having the boot components detached signed is the only OS for which it would be complicated to accokplish to provide the same security premises it could do tomorrow on all other OSes if the hook scripts were developed and deployed iteratively.

This is why I tagged you, @DemiMarie. As proposed, in this issue proposal: would QubesOS consider this approach, not another one, as valid because it would work for other OSes and would extend what Heads security premises can offer if we were to go in that direction without requiring anything else then what is proposed here.

@marmarek
Copy link
Contributor

Posting my message from today discussion:

I have an idea (feel free to ignore ;) ) to improve the message about /boot files changes - add a diagram to that error message, explaining which step went wrong, kinda like cloudflare error messages which show nice and easy where the problem is. It could be something like Nitrokey -> Heads -> boot config -> boot files (or "kernel" or "operating system"), and mark which verification failed. I guess it could be even ascii-art (with colors? if whiptail supports that) for simpler implementation / space saving. This should avoid confusion about "Nitrokey blinks green but Heads says something is wrong".

It's not a solution to the original proposal here (which would be much better), but it's a band-aid for the current situation.

@tlaurion
Copy link
Collaborator

Posting my message from today discussion:

I have an idea (feel free to ignore ;) ) to improve the message about /boot files changes - add a diagram to that error message, explaining which step went wrong, kinda like cloudflare error messages which show nice and easy where the problem is. It could be something like Nitrokey -> Heads -> boot config -> boot files (or "kernel" or "operating system"), and mark which verification failed. I guess it could be even ascii-art (with colors? if whiptail supports that) for simpler implementation / space saving. This should avoid confusion about "Nitrokey blinks green but Heads says something is wrong".

It's not a solution to the original proposal here (which would be much better), but it's a band-aid for the current situation.

Or @marmarek @nestire @daringer, if need is for hotp_verification to permit a red only toogle to fix the problem since UI to me is already pretty clear, maybe open an issue on hotp-verification side @marmarek ?

Per matrix discussion under qubesos-public channel, which happens all the time:

Not sure this fix would help if UX expects hotp security dongle led to flash led here

Image extracted from referred discussion:

ima_76730ef_20241220163607.jpeg

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
None yet
Projects
None yet
Development

No branches or pull requests

4 participants