├── live-build └── config │ ├── archives │ ├── experimental.list.chroot │ └── experimental.pref.chroot │ ├── hooks │ └── 0510-root-password.hook.chroot │ ├── includes.chroot │ ├── etc │ │ ├── firejail │ │ │ ├── firefox.profile │ │ │ └── blacklist-dev.inc │ │ ├── rc.local │ │ └── iceweasel │ │ │ └── profile │ │ │ ├── searchplugins │ │ │ ├── startpage-ssl.xml │ │ │ └── wikipedia-ssl.xml │ │ │ └── prefs.js │ └── lib │ │ └── live │ │ └── config │ │ └── 1200-setattr │ └── package-lists │ └── my.list.chroot ├── README.md └── GUIDE.md /live-build/config/archives/experimental.list.chroot: -------------------------------------------------------------------------------- 1 | deb http://ftp.debian.org/debian unstable main 2 | deb http://ftp.debian.org/debian experimental main 3 | -------------------------------------------------------------------------------- /live-build/config/hooks/0510-root-password.hook.chroot: -------------------------------------------------------------------------------- 1 | #!/bin/sh 2 | 3 | set -e 4 | 5 | usermod -p '$5$GxpbCxvGOmtudd$uL90C.ZOMY6WxI4.x32kTCv38dGiXYUlfWGzCQuHmr3' root 6 | 7 | -------------------------------------------------------------------------------- /live-build/config/archives/experimental.pref.chroot: -------------------------------------------------------------------------------- 1 | Package: iceweasel 2 | Pin: release a=experimental 3 | Pin-Priority: 995 4 | 5 | Package: * 6 | Pin: release a=unstable 7 | Pin-Priority: 1 8 | -------------------------------------------------------------------------------- /live-build/config/includes.chroot/etc/firejail/firefox.profile: -------------------------------------------------------------------------------- 1 | # Firejail profile for Mozilla Firefox (Iceweasel in Debian) 2 | include /etc/firejail/disable-mgmt.inc 3 | include /etc/firejail/disable-secret.inc 4 | include /etc/firejail/blacklist-dev.inc 5 | blacklist ${HOME}/.adobe 6 | blacklist ${HOME}/.macromedia 7 | caps 8 | seccomp 9 | -------------------------------------------------------------------------------- /live-build/config/includes.chroot/etc/rc.local: -------------------------------------------------------------------------------- 1 | #!/bin/sh -e 2 | # 3 | # rc.local 4 | # 5 | # This script is executed at the end of each multiuser runlevel. 6 | # Make sure that the script will "exit 0" on success or any other 7 | # value on error. 8 | # 9 | # In order to enable or disable this script just change the execution 10 | # bits. 11 | # 12 | # By default this script does nothing. 13 | 14 | -------------------------------------------------------------------------------- /live-build/config/package-lists/my.list.chroot: -------------------------------------------------------------------------------- 1 | task-english pm-utils alsa-utils unzip screen curl iceweasel spice-vdagent fluxbox attr xserver-xorg-video-modesetting xserver-xorg-video-qxl xserver-xorg-input-evdev xinit x11-apps x11-utils x11-xserver-utils rxvt-unicode iceweasel-noscript iceweasel-perspectives iceweasel-refcontrol iceweasel-requestpolicy iceweasel-openinbrowser iceweasel-https-everywhere 2 | -------------------------------------------------------------------------------- /live-build/config/includes.chroot/lib/live/config/1200-setattr: -------------------------------------------------------------------------------- 1 | #!/bin/sh 2 | 3 | Init () 4 | { 5 | echo -n " setattr" 6 | } 7 | 8 | Config () 9 | { 10 | if [ -f /usr/bin/iceweasel -a -f /usr/bin/setfattr ]; then 11 | setfattr -n user.pax.flags -v m /usr/bin/iceweasel 12 | setfattr -n user.pax.flags -v m /usr/lib/iceweasel/plugin-container 13 | fi 14 | 15 | # Creating state file 16 | touch /var/lib/live/config/setattr 17 | } 18 | 19 | Init 20 | Config 21 | -------------------------------------------------------------------------------- /live-build/config/includes.chroot/etc/firejail/blacklist-dev.inc: -------------------------------------------------------------------------------- 1 | seccomp mknod 2 | 3 | blacklist /dev/autofs 4 | blacklist /dev/block 5 | blacklist /dev/btrfs-control 6 | blacklist /dev/bus 7 | blacklist /dev/char 8 | blacklist /dev/console 9 | blacklist /dev/cpu 10 | blacklist /dev/cpu_dma_latency 11 | blacklist /dev/disk 12 | blacklist /dev/dri 13 | blacklist /dev/fb0 14 | blacklist /dev/fd 15 | blacklist /dev/full 16 | blacklist /dev/fuse 17 | blacklist /dev/grsec 18 | blacklist /dev/hpet 19 | blacklist /dev/hugepages 20 | blacklist /dev/initctl 21 | blacklist /dev/input 22 | blacklist /dev/kmsg 23 | blacklist /dev/log 24 | blacklist /dev/loop* 25 | blacklist /dev/mapper 26 | blacklist /dev/mcelog 27 | blacklist /dev/mem 28 | blacklist /dev/memory_bandwidth 29 | blacklist /dev/mqueue 30 | blacklist /dev/psaux 31 | blacklist /dev/ptmx 32 | blacklist /dev/pts 33 | blacklist /dev/rtc 34 | blacklist /dev/rtc0 35 | blacklist /dev/shm 36 | blacklist /dev/tty* 37 | blacklist /dev/uinput 38 | blacklist /dev/vcs* 39 | blacklist /dev/vda* 40 | blacklist /dev/vga_arbiter 41 | blacklist /dev/vhost-net 42 | blacklist /dev/virtio* 43 | blacklist /dev/vport* 44 | blacklist /dev/xconsole 45 | -------------------------------------------------------------------------------- /live-build/config/includes.chroot/etc/iceweasel/profile/searchplugins/startpage-ssl.xml: -------------------------------------------------------------------------------- 1 | 2 | Startpage (SSL) 3 | Startpage is a search engine that does not collect or share personally identifiable information. This search bar extension utilizes SSL to protect transmissions between You and Startpage. 4 | UTF-8 5 |  6 | https://startpage.com/ 7 | 8 | 9 | 10 | 11 | 12 | -------------------------------------------------------------------------------- /README.md: -------------------------------------------------------------------------------- 1 | # Live-Armor 2 | 3 | This repository contains the 4 | [Live-Armor Guide](https://fatemachine.github.io/live-armor/), a guide 5 | to building custom Linux 6 | [live images](https://en.wikipedia.org/wiki/Live_CD) for security 7 | sandboxing using tools from the [Debian](https://www.debian.org) 8 | [Live Systems](http://www.live-systems.org/) project and 9 | [Grsecurity](https://www.grsecurity.net/). 10 | 11 | The `live-build` directory contains an example configuration for Debian 12 | [live-build](http://live-systems.org/manual/current/html/live-manual/installation.en.html#118) 13 | that can be used as a starting point for building a custom live image. 14 | This configuration is based on the one covered in the Guide. 15 | 16 | ## Quick Start 17 | 18 | 1. Install [live-build 19 | 5.0](https://packages.debian.org/experimental/live-build). 20 | 21 | 1. Create an empty directory that will contain your live image 22 | configuration and build data. 23 | 24 | 1. Change to your live image directory and run: `lb config` 25 | 26 | 1. Copy the `live-build/config` tree of this repository into the 27 | `config` subdirectory that `lb config` just created, for example by 28 | using `cp -r`. 29 | 30 | 1. Edit `config/binary` and add the `union=overlay` and optionally 31 | `live-config.noroot` kernel boot parameters to the 32 | `LB_BOOTAPPEND_LIVE` and `LB_BOOTAPPEND_LIVE_FAILSAFE` variables. 33 | 34 | If you added `live-config.noroot` to disable sudo, choose your root 35 | password by running `mkpasswd` (part of the 36 | [whois](https://packages.debian.org/whois) package) and replace the 37 | argument to `usermod -p` in 38 | `config/hooks/0510-root-password.hook.chroot` with the output of 39 | `mkpasswd`. 40 | 41 | If you did not add `live-config.noroot` and are therefore using the 42 | default unprotected sudo access method, delete 43 | `config/hooks/0510-root-password.hook.chroot`. 44 | 45 | Note: The default login credentials are username `user` and password 46 | `live`. 47 | 48 | 1. Edit `config/chroot` and change `LB_UNION_FILESYSTEM="aufs"` to 49 | `LB_UNION_FILESYSTEM="overlay"`. 50 | 51 | 1. Inspect the list of custom packages in 52 | `config/package-lists/my.list.chroot` and make any desired changes. 53 | 54 | 1. Add `.deb` package files to the `config/packages.chroot` directory: 55 | 56 | * Your custom live system kernel package, with a name that begins 57 | with `linux-image`. See the Guide for instructions on configuring 58 | and building a custom kernel. 59 | * `live-boot` and `live-boot-initramfs-tools` packages patched for 60 | OverlayFS support. See the Guide for instructions. 61 | * `live-config` and `live-config-systemd` package versions that match 62 | the `live-boot` version, for example from Debian `experimental`. 63 | 64 | 1. Re-run `lb config`. 65 | 66 | 1. Build your live image: `lb build 2>&1 | tee build.log` 67 | 68 | This configuration has been tested with the following software versions: 69 | 70 | - Debian [jessie](https://www.debian.org/releases/jessie/) 71 | - Linux [kernel](https://www.kernel.org/) version 3.18.x 72 | - [live-build](https://packages.debian.org/live-build) 5.0a3 73 | - [live-boot](https://packages.debian.org/live-boot) and 74 | [live-config](https://packages.debian.org/live-config) 5.0a1, with 75 | live-boot patched for OverlayFS support 76 | - [QEMU](https://packages.debian.org/qemu-system-x86) 2.1 77 | -------------------------------------------------------------------------------- /live-build/config/includes.chroot/etc/iceweasel/profile/searchplugins/wikipedia-ssl.xml: -------------------------------------------------------------------------------- 1 | 2 | Wikipedia (SSL) 3 | Wikipedia is an openly-editable, free content internet encyclopedia. This search bar add-on utilizes SSL to protect searches between You and Wikipedia. 4 | UTF-8 5 |  6 | https://en.wikipedia.org/wiki/Special:Search 7 | 8 | 9 | 10 | 11 | 12 | 13 | -------------------------------------------------------------------------------- /live-build/config/includes.chroot/etc/iceweasel/profile/prefs.js: -------------------------------------------------------------------------------- 1 | # Mozilla User Preferences 2 | 3 | /* Don't start finding text when any input is typed */ 4 | user_pref("accessibility.typeaheadfind.autostart", false); 5 | 6 | /* Start with a simple blank page */ 7 | user_pref("browser.startup.page", 1); 8 | user_pref("browser.startup.homepage", "about:about"); 9 | 10 | /* Don't send every URL we visit to Google */ 11 | user_pref("browser.safebrowsing.enabled", false); 12 | user_pref("browser.safebrowsing.malware.enabled", false); 13 | 14 | /* Don't save information entered in web forms and search bars */ 15 | user_pref("browser.formfill.enable", false); 16 | 17 | /* Backspace goes back one page in history */ 18 | user_pref("browser.backspace_action", 0); 19 | 20 | /* Ask where to save downloads */ 21 | user_pref("browser.download.useDownloadDir", false); 22 | 23 | /* When JavaScript wants to open a new window, open a tab instead */ 24 | user_pref("browser.link.open_newwindow.restriction", 0); 25 | 26 | /* Always use Private Browsing Mode */ 27 | user_pref("browser.privatebrowsing.autostart", true); 28 | 29 | /* Disable search suggestions */ 30 | user_pref("browser.search.suggest.enabled", false); 31 | 32 | /* Limit state information saved between sessions */ 33 | user_pref("browser.sessionstore.max_tabs_undo", 5); 34 | user_pref("browser.sessionstore.resume_from_crash", false); 35 | 36 | /* Turn off tab animations */ 37 | user_pref("browser.tabs.animate", false); 38 | 39 | /* Turn off URL bar funny business */ 40 | user_pref("browser.urlbar.autocomplete.enabled", false); 41 | user_pref("browser.urlbar.trimURLs", false); 42 | 43 | /* Disallow JavaScript access to potentially dangerous APIs */ 44 | user_pref("dom.event.clipboardevents.enabled", false); 45 | user_pref("dom.battery.enabled", false); 46 | user_pref("dom.disable_window_open_feature.menubar", true); 47 | user_pref("dom.disable_window_open_feature.personalbar", true); 48 | user_pref("dom.disable_window_open_feature.scrollbars", true); 49 | user_pref("dom.disable_window_open_feature.toolbar", true); 50 | user_pref("dom.popup_maximum", 10); 51 | user_pref("dom.storage.default_quota", 0); 52 | 53 | /* Override User-Agent data to mitigate browser fingerprinting. 54 | * See https://panopticlick.eff.org/ 55 | */ 56 | user_pref("general.appname.override", "Netscape"); 57 | user_pref("general.appversion.override", "5.0 (Windows)"); 58 | user_pref("general.buildID.override", 0); 59 | user_pref("general.oscpu.override", "Windows NT 6.2"); 60 | user_pref("general.platform.override", "Win32"); 61 | user_pref("general.productSub.override", "20100101"); 62 | user_pref("general.useragent.override", "Mozilla/5.0 (Windows NT 6.2; rv:36.0) Gecko/20100101 Firefox/36.0"); 63 | user_pref("general.useragent.vendor", ""); 64 | user_pref("general.useragent.vendorSub", ""); 65 | user_pref("general.warnOnAboutConfig", false); 66 | user_pref("intl.accept_languages", "en-us,en;q=0.5"); 67 | 68 | /* Turn off "location aware browsing" */ 69 | user_pref("geo.enabled", false); 70 | 71 | /* Turn off sending non-URL words entered in the URL bar to Google */ 72 | user_pref("keyword.enabled", false); 73 | 74 | /* Turn off spell-checking */ 75 | user_pref("layout.spellcheckDefault", 0); 76 | 77 | /* Disable cookies by default. Use an extension for site-specific 78 | * whitelisting. 79 | */ 80 | user_pref("network.cookie.cookieBehavior", 2); 81 | user_pref("network.cookie.thirdparty.sessionOnly", true); 82 | 83 | /* Disable IPv6 unless you want to use it. */ 84 | user_pref("network.dns.disableIPv6", true); 85 | 86 | /* Don't "proactively" perform DNS resolution */ 87 | user_pref("network.dns.disablePrefetch", true); 88 | 89 | /* Unneeded unless we're using v6 */ 90 | user_pref("network.http.fast-fallback-to-IPv4", false); 91 | 92 | /* Enable pipelining for better performance */ 93 | user_pref("network.http.pipelining", true); 94 | user_pref("network.http.pipelining.maxrequests", 15); 95 | user_pref("network.http.pipelining.ssl", true); 96 | user_pref("network.http.proxy.pipelining", true); 97 | user_pref("network.http.redirection-limit", 5); 98 | 99 | /* Don't "proactively" fetch pages that haven't been requested */ 100 | user_pref("network.prefetch-next", false); 101 | 102 | /* Disable Websockets by default */ 103 | user_pref("network.websocket.enabled", false); 104 | 105 | /* Enable the Do-Not-Track header in HTTP requests */ 106 | user_pref("privacy.donottrackheader.enabled", true); 107 | 108 | /* Clear Private Data when closing the browser */ 109 | user_pref("privacy.sanitize.sanitizeOnShutdown", true); 110 | 111 | /* Disable unsafe RC4 ciphers */ 112 | user_pref("security.ssl3.ecdhe_ecdsa_rc4_128_sha", false); 113 | user_pref("security.ssl3.ecdhe_rsa_rc4_128_sha", false); 114 | user_pref("security.ssl3.rsa_rc4_128_md5", false); 115 | user_pref("security.ssl3.rsa_rc4_128_sha", false); 116 | 117 | /* Disable WebGL by default */ 118 | user_pref("webgl.disabled", true); 119 | -------------------------------------------------------------------------------- /GUIDE.md: -------------------------------------------------------------------------------- 1 | # The Live-Armor Guide 2 | ## Building Custom Live Images with Debian and Grsecurity 3 | 4 | This guide explains how to build custom live system images for security 5 | sandboxing using tools from the [Debian](https://www.debian.org) 6 | [Live Systems](http://www.live-systems.org/) project and 7 | [Grsecurity](https://www.grsecurity.net). 8 | 9 | For concreteness we will focus on building a custom live image for 10 | sandboxing the Firefox web browser (also known as Iceweasel in the Debian 11 | world). However, the same tools and procedures will allow you to build any 12 | kind of Debian-based live image you want. 13 | 14 | - [Motivation](#motivation) 15 | - [Prerequisites](#prerequisites) 16 | - [Limitations and Alternatives](#limitations-and-alternatives) 17 | - [Architecture](#architecture) 18 | - [Layer 1: The Host System](#layer-1-the-host-system) 19 | - [Layer 2: The Guest Image](#layer-2-the-guest-image) 20 | - [Layer 3: Firejail](#layer-3-firejail) 21 | - [Layer 4: Firefox/Iceweasel Customization](#layer-4-firefoxiceweasel-customization) 22 | 23 | ## Motivation 24 | 25 | ### The browser problem 26 | 27 | If you're alive and technically aware in 2015, you know that Internet, 28 | operating system, and applications software are security disaster areas. 29 | End-user systems such as workstations and laptops are especially hard to 30 | protect against the growing tide of malware, most of it delivered via the 31 | Internet. 32 | 33 | After basic security measures have been taken, such as disabling unnecessary 34 | operating system services and firewalling network access, the largest attack 35 | surface on a typical end-user system today is the web browser. Popular 36 | browsers are large and complex pieces of software, invariably written in 37 | [unsafe](https://en.wikipedia.org/wiki/Memory_safety) languages for 38 | performance (although this may slowly be 39 | [changing](https://github.com/servo/servo)). 40 | [Critical](https://www.mozilla.org/en-US/security/advisories/) 41 | [vulnerabilities](http://www.cvedetails.com/vulnerability-list/vendor_id-1224/product_id-15031/opec-1/Google-Chrome.html) 42 | are discovered in popular browsers every month, any one of which can allow a 43 | remote attacker to take complete control over the computer running the 44 | browser. While this problem is now well known and recognized by browser 45 | vendors and efforts are being made to improve the status quo, building a 46 | secure browser has turned out to be a long and difficult road. Meanwhile, 47 | users are left adrift and vulnerable, and even the technically skilled have 48 | few good options for securing web browser installations. 49 | 50 | ### Live images 51 | 52 | A *live image* is a complete operating system image file (usually in 53 | [ISO format](https://en.wikipedia.org/wiki/ISO_image)) that can be loaded at 54 | system boot time from CD/DVD/USB media and that runs completely from RAM. 55 | No hard disk or any other form of persistent media is required. Changes can 56 | be made normally to the running system, but all changes are lost when the 57 | system is powered down or reset. (Some live images support an optional 58 | persistence partition that allows the user to store data that persists 59 | across restarts.) 60 | 61 | Live images are a powerful tool for 62 | [sandboxing](https://en.wikipedia.org/wiki/Sandbox_(computer_security)). 63 | Even if the live operating system or applications are compromised during 64 | runtime, the next reboot will restore the system to its original clean 65 | state. Simply using a live system is no guarantee of security, but it can 66 | make long-term compromise of a system significantly more difficult, 67 | especially when combined with other countermeasures as part of a 68 | top-to-bottom architecture that takes into account each layer from the 69 | lowest (physical setting) through the highest (user applications). 70 | 71 | ## Prerequisites 72 | 73 | This guide assumes you are an experienced systems administrator who is 74 | comfortable with Debian and with configuring and building the 75 | [Linux kernel](https://www.kernel.org/). 76 | 77 | ## Limitations and Alternatives 78 | 79 | The live image method is a relatively heavyweight approach to browser 80 | security. Configuring and building a custom live image takes time and 81 | skill, and using one is harder than using a browser directly. All 82 | browser customization must be done as part of live system configuration, 83 | which means that making even simple persistent changes to the browser 84 | requires updating the live configuration and rebuilding the live image. 85 | This is more work than most users are willing to put up with. 86 | 87 | There are a few alternatives: 88 | 89 | - Use a pre-built live system. This obviates the need to configure and 90 | build the live image yourself, and is ideal if you can find such an image 91 | that you trust and that closely matches your needs. Unfortunately, few 92 | such images seem to be available today that are built with security in 93 | mind. An exception is [Tails](https://tails.boum.org/), but it may not be 94 | suitable unless you want to route all data through Tor. 95 | 96 | - Use a [chroot](https://en.wikipedia.org/wiki/Chroot) jail. Setting up a 97 | secure chroot environment is a difficult task, and should only be 98 | attempted if you are fully aware of the security weaknesses of vanilla 99 | chroot environments and employ kernel hardening measures to guard against 100 | them, such as the chroot restrictions available in 101 | [Grsecurity](https://www.grsecurity.net/) kernels. A chroot jail provides 102 | a relatively thin barrier between the guest and host environments and does 103 | not in itself provide the "clean boot" property of a live image. 104 | 105 | - Use a [namespace/cgroups](https://en.wikipedia.org/wiki/Cgroups) jail. 106 | Newer Linux kernels provide a range of other features that can be used to 107 | create container or jail environments. As with chroot, such environments 108 | are difficult to configure securely, although tools such as 109 | [Firejail](https://l3net.wordpress.com/projects/firejail/) can make this 110 | easier. And as with chroot jails, programs running in a namespace jail 111 | can still make system calls directly into the host kernel, and thus are in 112 | a position to exploit kernel-level security flaws. 113 | 114 | - Use a kernel security framework like 115 | [AppArmor](https://en.wikipedia.org/wiki/AppArmor), 116 | [SELinux](https://en.wikipedia.org/wiki/SELinux), or Grsecurity 117 | [RBAC](https://en.wikibooks.org/wiki/Grsecurity/The_RBAC_System) to 118 | restrict the areas of the system that the application can access. These 119 | frameworks serve mainly to prevent a compromised program from affecting 120 | the rest of the system. They are best used along with other exploit 121 | prevention and sandboxing measures. 122 | 123 | ## Architecture 124 | 125 | In this guide we will assume a four-layer architecture that comprises: 126 | 127 | 1. A host system, such as a workstation or laptop 128 | 1. A guest live system, running as a virtual machine on the host system 129 | 1. A container jail environment, running inside the guest system 130 | 1. A Firefox/Iceweasel browser, running inside the jail 131 | 132 | We could consider variations, such as booting and running the live image on 133 | bare metal. In fact, most of this guide concerns setting up the live image 134 | and applies equally well regardless of how and where the live system is run. 135 | One of the advantages of a live image is that it can be carried around on 136 | bootable read-only media and run from any system you happen to have access 137 | to, provided the image was built with the drivers needed to drive the host 138 | hardware. 139 | 140 | A word on virtual machines. A virtual machine can never be more secure than 141 | the host it's running on. Even if your guest operating system and 142 | applications are completely secure, all is lost if the host has been 143 | compromised. The purpose of building a live image sandbox is to protect the 144 | host from the guest and its applications. There is no way to protect the 145 | guest from the host. 146 | 147 | Running applications like web browsers within a virtual machine provides a 148 | relatively high degree of isolation between them and the host. If the 149 | application is compromised, the attacker may gain control over the guest 150 | system, but attacking the host from the guest is a difficult task as long as 151 | basic host security precautions have been taken. Although guest-to-host 152 | exploits that attack the hypervisor interface are 153 | [possible](https://lwn.net/Articles/619376/), such hypervisor flaws are rare 154 | compared to browser flaws and the hypervisor attack surface is much smaller. 155 | The risk can be mitigated further through 156 | [kernel hardening](https://en.wikipedia.org/wiki/PaX) on both the host and 157 | guest systems, and by using a kernel security framework on the host to 158 | restrict the access of hypervisor processes. 159 | 160 | The architecture described in this guide is an example of the principle of 161 | [defense in depth](https://en.wikipedia.org/wiki/Defense_in_depth_(computing)). 162 | Multiple independent security layers are employed simultaneously, each 163 | providing a qualitatively different containment barrier around the 164 | vulnerable application. These security layers are: 165 | 166 | 1. A host system with a kernel hardened using PaX/Grsecurity, and 167 | restriction of hypervisor processes using Grsecurity Role Based Access 168 | Control (RBAC) 169 | 1. A guest system built as a read-only live image with a PaX kernel 170 | 1. A container jail environment inside the guest, created with 171 | [Firejail](https://l3net.wordpress.com/projects/firejail/) 172 | 1. Security-oriented application-level configuration of a Firefox browser, 173 | including default settings and extensions 174 | 175 | ## Layer 1: The Host System 176 | 177 | For our purposes, the only role of the host system is to provide an 178 | environment for building and running a live image. We will assume that the 179 | host is a Linux system and that [KVM](http://linux-kvm.org/) will be used to 180 | run the live image. 181 | 182 | ### Step 1.1: Configure PaX/Grsecurity 183 | 184 | This step is optional, but recommended. Running a Linux kernel patched with 185 | PaX/Grsecurity on the host provides a significantly better security baseline 186 | than vanilla Linux. The Grsecurity RBAC system also provides a way to 187 | restrict the access privileges of hypervisor (QEMU/KVM) processes and other 188 | programs running on the host. 189 | 190 | Setting up Grsecurity requires downloading the Linux kernel 191 | [source](https://www.kernel.org/), applying the 192 | [Grsecurity patch](https://www.grsecurity.net/) that matches the kernel 193 | version, configuring the kernel (including PaX/Grsecurity parameters), 194 | building the kernel, and finally installing and booting it. Assuming 195 | you have downloaded the kernel source as a file with a name like 196 | `linux-x.y.z.tar.xz` and the corresponding Grsecurity 197 | [patch](https://www.grsecurity.net/download.php) as `grsecurity.patch` 198 | (the actual patch filename contains version and date information), the 199 | basic steps are: 200 | 201 | ``` 202 | $ tar axf linux-x.y.z.tar.xz 203 | $ cd linux-x.y.z 204 | $ patch -p1 < ../grsecurity.patch 205 | $ make nconfig 206 | [edit and save kernel configuration] 207 | $ make deb-pkg 208 | ``` 209 | 210 | If all goes well, this will produce a `linux-image` Debian package file 211 | along with related package files for kernel headers, firmware, 212 | etc. according to Debian kernel-packaging conventions. These `.deb` 213 | files can be installed directly with `dpkg -i`. Another way to build 214 | Debian packages from kernel source is to use 215 | [kernel-package](https://packages.debian.org/kernel-package). 216 | 217 | Refer to the 218 | [Grsecurity documentation](https://en.wikibooks.org/wiki/Grsecurity) for 219 | instructions on applying the Grsecurity patch and configuring the 220 | PaX/Grsecurity kernel settings. When configuring the kernel, try to 221 | eliminate any drivers and features that you don't need on the host system, 222 | and enable other important security options such as 223 | [module signing](https://www.kernel.org/doc/Documentation/module-signing.txt), 224 | [stack protection](https://lwn.net/Articles/584225/), and 225 | [seccomp](https://wiki.mozilla.org/Security/Sandbox/Seccomp). Consult the 226 | Ubuntu 227 | [Kernel Hardening](https://wiki.ubuntu.com/Security/Features#Kernel_Hardening) 228 | feature checklist for other important kernel security features and 229 | parameters. 230 | 231 | ### Step 1.2: Configure the hypervisor 232 | 233 | Ensure QEMU/KVM is installed on the host: 234 | 235 | ``` 236 | # apt-get install qemu-kvm 237 | ``` 238 | 239 | For QEMU/KVM, hypervisor configuration takes the form of passing a set of 240 | command-line options to `kvm` (which itself is just a wrapper around 241 | `qemu-system-x86_64`). Here is an example script that can be used to boot 242 | an image with some useful QEMU options: 243 | 244 | ```shell 245 | #!/bin/sh 246 | 247 | export QEMU_AUDIO_DRV=alsa 248 | 249 | exec /usr/bin/kvm -cpu host -m 2048 -drive file=$1,if=virtio,media=cdrom \ 250 | -balloon virtio \ 251 | -usbdevice tablet \ 252 | -soundhw hda \ 253 | -vga std \ 254 | -netdev user,id=network0 -device virtio-net,netdev=network0 \ 255 | -virtfs local,path=/tmp/guest_share,mount_tag=share,security_model=mapped-xattr 256 | ``` 257 | 258 | To use this script, pass the image you wish to boot as an argument. 259 | 260 | * `-cpu host` tells QEMU to emulate the precise CPU that the host uses, 261 | rather than some different or more generic CPU. This provides the best 262 | performance. 263 | * `-m 2048` allocates 2GB of RAM to the guest (the default is 128MB). 264 | * `-drive file=$1,if=virtio,media=cdrom` says to use the filename passed as 265 | the argument `$1` as the virtual boot drive, to treat it as CDROM media, 266 | and to represent it as a [Virtio](http://www.linux-kvm.org/page/Virtio) 267 | device rather than emulating some form of disk hardware. This requires 268 | Virtio driver support in the guest kernel. Use Virtio interfaces and 269 | drivers whenever possible for best performance and feature support. 270 | * `-balloon virtio` enables 271 | [balloon](http://www.linux-kvm.org/page/Projects/auto-ballooning) support, 272 | allowing the guest to dynamically grow and release memory back to the 273 | host. This requires memory balloon support in the guest kernel. 274 | * `-usbdevice tablet` is necessary to prevent mismatched host/guest mouse 275 | pointers when using VNC. 276 | * `-soundhw hda` instructs QEMU to emulate HDA PCI sound hardware. Combined 277 | with the `export QEMU_AUDIO_DRV=alsa` line, this should provide working 278 | guest-to-host sound. If you don't need sound, you can remove these two 279 | lines. This requires HDA sound driver support in the guest kernel. 280 | * `-vga std` enables high-resolution video mode support. 281 | * `-netdev user,id=network0 -device virtio-net,netdev=network0` enables 282 | basic networking support, placing the guest on a private virtual network 283 | that can access the Internet via the host. 284 | * `-virtfs local,path=/tmp/guest_share,mount_tag=share,security_model=mapped-xattr` 285 | sets up a shared directory between the host and the guest so that files 286 | can easily be moved back and forth. The directory `/tmp/guest_share` on 287 | the host is made accessible to the guest using 288 | [9p](http://www.linux-kvm.org/page/9p_virtio). This requires 9p Virtio 289 | support in the guest kernel. 290 | 291 | You should verify that you can use a script like the above to boot a vanilla 292 | live ISO image on the host. Standard Debian 293 | [live images](http://cdimage.debian.org/debian-cd/current-live/amd64/iso-hybrid/) 294 | are available from the [Live Systems project](http://live-systems.org/). 295 | 296 | To use a VNC display rather than QEMU's default SDL display, you can add an 297 | option like `-vnc 127.0.0.1:0` and use a VNC client to connect to QEMU on 298 | port 5900. 299 | 300 | There is a newer display technology for QEMU called 301 | [SPICE/QXL](http://www.linux-kvm.org/page/SPICE). In theory, this should 302 | provide better graphics capabilities and performance, and also provides a 303 | mechanism for copy/paste clipboard transfer between the guest and host. In 304 | testing on a Linux 3.18 host/guest with QEMU 2.1, however, using QXL led to 305 | guest hangs with the QEMU host process consuming 100% CPU when doing even 306 | basic browsing. To enable SPICE/QXL you can use options like: 307 | 308 | ``` 309 | -vga qxl \ 310 | -spice addr=127.0.0.1,port=5910,disable-ticketing,image-compression=off 311 | ``` 312 | 313 | You will need a [SPICE client](https://packages.debian.org/spice-client) on 314 | the host to connect to the guest display. You will also need to enable QXL 315 | in the guest kernel and install the 316 | [QXL Xorg driver](https://packages.debian.org/xserver-xorg-video-qxl) in the 317 | guest. For copy/paste clipboard support, you will also need the 318 | [spice-vdagent](https://packages.debian.org/spice-vdagent) package in the 319 | guest. See the SPICE documentation for details. 320 | 321 | Another option to consider is `-sandbox on`. This enables 322 | [seccomp](https://wiki.mozilla.org/Security/Sandbox/Seccomp) sandboxing of 323 | the QEMU process. Unfortunately, in testing with Linux 3.18 and QEMU 2.1 324 | this led to instability of QEMU, so use this option with care. 325 | 326 | ### Step 1.3: Secure the Hypervisor 327 | 328 | The purpose of this step is to lock down the access that the hypervisor 329 | process has to the rest of the system. Apart from read/write access to 330 | `/dev/kvm`, QEMU has no special access requirements and can safely be 331 | prevented from reading, writing, or executing most filesystem paths. 332 | Its network access can also be restricted to whatever is required for 333 | the applications you run within the VM. 334 | 335 | The procedure for enforcing access control depends on the security 336 | framework you're using on the host. Most frameworks have a *learning 337 | mode* that allows you to run an application and exercise its 338 | functionality, and that then generates a whitelist policy that permits 339 | access to only those resources actually used by the application during 340 | the learning process. Usually some degree of manual review and tuning 341 | of the generated access policy is needed. Arriving at a good policy 342 | usually requires several tweaks and iterations. 343 | 344 | For example, if you're using the Grsecurity RBAC system and QEMU-x86_64, 345 | you can start with a stub entry in your policy file under the 346 | appropriate user role that looks like this: 347 | 348 | ``` 349 | subject /usr/bin/qemu-system-x86_64 lo { 350 | / 351 | } 352 | ``` 353 | 354 | The `l` flag places the policy for this program in learning mode. You 355 | can then use `gradm` as usual to enable RBAC in learning mode, exercise 356 | the program, and then generate an initial policy. See the 357 | [RBAC documentation](https://en.wikibooks.org/wiki/Grsecurity/The_RBAC_System) 358 | for details. 359 | 360 | Although Grsecurity RBAC is powerful, it uses a whole-system whitelist 361 | model, meaning that using it requires having a policy that covers all 362 | users and programs on the system. In spite of its 'full system 363 | learning' mode, generating a working full-system poilcy is a difficult 364 | and time-consuming exercise, especially for desktop-oriented systems and 365 | those that depend on invasive multi-function services such as systemd. 366 | A far simpler, though less comprehensive, alternative is 367 | [AppArmor](https://en.wikipedia.org/wiki/AppArmor), which only restricts 368 | programs that have policies defined and has no concept of users or 369 | roles. 370 | 371 | ## Layer 2: The Guest Image 372 | 373 | Now that the host system has been prepared and QEMU is working, we are 374 | ready to configure and build the live image. 375 | 376 | ### Step 2.1: Configuring and Building the Guest Kernel 377 | 378 | Follow the same procedure used to configure and build the 379 | [host kernel](#step-11-configure-paxgrsecurity). When configuring the 380 | guest kernel, note the following points: 381 | 382 | - Remove all hardware drivers except for devices that QEMU can emulate 383 | and that you require. For example, if you want sound support in the 384 | VM, configure the guest kernel to support Intel PCI HDAudio and start 385 | QEMU with the sound options given above. You do not need 386 | hardware-specific drivers for disk, network, or graphics as these will 387 | be handled by Virtio — see below. 388 | - Enable basic virtualization options such as: 389 | * `CONFIG_PARAVIRT` 390 | * `CONFIG_HYPERVISOR_GUEST` 391 | * `CONFIG_KVM_GUEST` 392 | - Many core devices such as block and network devices have optimized 393 | *Virtio* drivers designed specifically for virtual machines. Whenever 394 | possible, use a Virtio driver instead of a hardware driver for a 395 | device that QEMU will have to emulate. Ensure the following options 396 | are enabled: 397 | * `CONFIG_VIRTIO_PCI` 398 | * `CONFIG_VIRTIO_BLK` 399 | * `CONFIG_VIRTIO_NET` 400 | * `CONFIG_HW_RANDOM_VIRTIO` 401 | * `CONFIG_VIRTIO_BALLOON` 402 | - Enable virtualized graphics options: 403 | * `CONFIG_DRM_CIRRUS_QEMU` 404 | * `CONFIG_DRM_QXL` 405 | * `CONFIG_DRM_BOCHS` 406 | - Enable OverlayFS support. This is **required** for a working live 407 | image: 408 | * `CONFIG_OVERLAY_FS` 409 | - If you want to be able to share host directories with the guest to 410 | easily move files back and forth, enable 9p support: 411 | * `CONFIG_NET_9P` 412 | * `CONFIG_NET_9P_VIRTIO` 413 | * `CONFIG_9P_FS` 414 | 415 | ### Step 2.2: Installing the Live-Build Tools on the Host 416 | 417 | The only package you need to build live images is 418 | [live-build](https://packages.debian.org/live-build). Several other 419 | packages such as [live-boot](https://packages.debian.org/live-boot) and 420 | [live-config](https://packages.debian.org/live-config) will be installed 421 | automatically into the live image during the build process. Full 422 | documentation on these packages is available from the 423 | [Live Systems project](http://live-systems.org/). 424 | 425 | To get started with `live-build`, create a directory that will be used 426 | to store your live image configuration and build data (it should have at 427 | least 1 GB of free space available), and then run `lb config` to create 428 | a default live image configuration: 429 | 430 | ``` 431 | $ mkdir my_live_image 432 | $ cd my_live_image 433 | $ lb config 434 | [2015-03-02 14:41:16] lb config 435 | P: Creating config tree for a debian/jessie/amd64 system 436 | P: Symlinking hooks... 437 | ``` 438 | 439 | This creates a configuration directory tree that live-build uses to 440 | determine how to build the live image. If you don't make any changes to 441 | this tree, you will end up with a default Debian Live image. You should 442 | try to build this default image now to ensure your system is set up 443 | correctly: 444 | 445 | ``` 446 | # lb build 2>&1 | tee build.log 447 | ``` 448 | 449 | This command will run for some time and generate a lot of output as it 450 | downloads packages, builds the system in a chroot environment, and 451 | finally freezes the chroot tree as a 452 | [squashfs](https://en.wikipedia.org/wiki/SquashFS) image and combines it 453 | with a bootloader and kernel into a bootable binary `.iso` image file. 454 | This final file will be placed in the directory where you ran `lb 455 | build`. 456 | 457 | Note: `lb build` expects to be run as root. If the build system is 458 | running a Grsecurity kernel, you will also have to temporarily 459 | deactivate some Grsecurity chroot restrictions for the build to succeed, 460 | e.g. with 461 | 462 | ``` 463 | # echo 0 > /proc/sys/kernel/grsecurity/chroot_caps 464 | # echo 0 > /proc/sys/kernel/grsecurity/chroot_deny_chmod 465 | # echo 0 > /proc/sys/kernel/grsecurity/chroot_deny_mount 466 | # echo 0 > /proc/sys/kernel/grsecurity/chroot_deny_mknod 467 | ``` 468 | 469 | #### OverlayFS Support 470 | 471 | Live images require some form of 472 | [Union filesystem](https://en.wikipedia.org/wiki/Union_filesystem) 473 | support in the guest kernel. Traditionally the Debian Live System tools 474 | have used [AuFS](http://aufs.sourceforge.net/) for this. Unfortunately, 475 | AuFS is an out-of-tree kernel patch that does not play nicely with 476 | Grsecurity. There is now an in-tree alternative called OverlayFS, 477 | enabled with the `CONFIG_OVERLAY_FS` option in the guest kernel. 478 | However, at the time of writing, the Debian `live-boot` package does not 479 | properly support OverlayFS. Debian 480 | [bug 773881](https://bugs.debian.org/773881) has been opened for this. 481 | In the bug comments, a link is provided to a Git repository and branch 482 | that contains a patched version of live-boot 5.0a1. Until the official 483 | live-boot package is updated, a working live-boot that supports 484 | OverlayFS can be built from this branch using `dpkg-buildpackage -b`. 485 | This will produce the files `live-boot_5.0~a1-1_all.deb` and 486 | `live-boot-initramfs-tools_5.0~a1-1_all.deb`. These packages should be 487 | placed in the `config/packages.chroot` directory when configuring the 488 | live image (see next step) along with version 5.0a1 of live-config, 489 | overriding the default live-boot and live-config packages that would 490 | otherwise be installed when building the live image. 491 | 492 | ### Step 2.3: Customizing the Live Image 493 | 494 | You have now used `lb build` to successfully build a default Debian Live 495 | ISO image. This image should be bootable and functional on virtual 496 | machines or real hardware. This step covers how to customize the live 497 | image. 498 | 499 | The general procedure for making changes to your live image is to modify 500 | some files in your live configuration tree (the directory where you ran 501 | `lb config`) and then rebuild the live image by running the following 502 | sequence of commands: 503 | 504 | ``` 505 | # lb clean 506 | # lb config 507 | # lb build 508 | ``` 509 | 510 | The `lb clean` command cleans up data from the last build, the `lb 511 | config` command updates the live image configuration based on your 512 | changes, and the `lb build` command actually builds the new image. 513 | 514 | From now on, we will assume that you are in your live-build root 515 | directory (the directory where you ran `lb config`). Unless otherwise 516 | noted, path names will be given relative to this directory. Most files 517 | and directories used for customization live under the `config/` 518 | subdirectory. 519 | 520 | #### Overview of Live Image Configuration 521 | 522 | See the 523 | [Managing a configuration](http://live-systems.org/manual/current/html/live-manual/managing-a-configuration.en.html) 524 | section of the Live Systems manual for an overview of live image 525 | configuration. Briefly, many basic options, like the target Debian 526 | distribution and architecture, can be specified as options to `lb 527 | config` — see `lb config --help` and `man lb_config` for details. 528 | Passing options to `lb config` results in modifications to one or more 529 | config files located in the `config` subdirectory of your build root. 530 | 531 | Although the manual recommends using auto scripts to manage a live image 532 | configuration, and for good reason, for simplicity we will make changes 533 | by editing configuration files directly. These changes will be 534 | preserved on subsequent runs of `lb config`. Be aware of the caveats of 535 | this approach as described in the manual and consider using auto scripts 536 | instead where possible. 537 | 538 | #### Choosing Packages to Install 539 | 540 | There are two ways to add packages of your choice to your live image. 541 | 542 | ##### Adding standard Debian packages 543 | 544 | For standard Debian packages, create a file with a name ending in 545 | `.list.chroot` in the `config/package-lists` directory. This file may 546 | contain a space-separated list of Debian package names. The named 547 | packages will be retrieved and installed in the chroot during the 548 | build/bootstrap process. 549 | 550 | Here is an example package list for a Firefox/Iceweasel image: 551 | 552 | ``` 553 | $ cat config/package-lists/my.list.chroot 554 | task-english pm-utils alsa-utils unzip screen curl iceweasel 555 | spice-vdagent fluxbox attr xserver-xorg-video-modesetting 556 | xserver-xorg-video-qxl xserver-xorg-input-evdev xserver-xorg-input-mouse 557 | xinit x11-apps x11-utils x11-xserver-utils rxvt-unicode 558 | iceweasel-noscript iceweasel-perspectives iceweasel-refcontrol 559 | iceweasel-requestpolicy iceweasel-openinbrowser 560 | iceweasel-https-everywhere 561 | ``` 562 | 563 | Remarks on some of these packages: 564 | 565 | - `attr` is required to set PaX flags on executables, assuming the 566 | `CONFIG_PAX_XATTR_PAX_FLAGS` kernel option was chosen (recommended). 567 | - `pm-utils` is included for `pm-suspend` which can be used to put the 568 | system to sleep temporarily. Sometimes suspending the VM is 569 | necessary before suspending the host to avoid a frozen VM when the 570 | host resumes. 571 | - `alsa-utils` is included for basic sound utilities such as 572 | `alsamixer`. 573 | - `spice-vdagent` is required for copy/paste support via QEMU SPICE, 574 | discussed earlier. 575 | - `fluxbox` is a lightweight window manager for X. You can choose any 576 | other window manager or desktop environment you like. 577 | - The `xserver-xorg` packages select X drivers important for QEMU, 578 | corresponding to the kernel drivers selected earlier. 579 | - The `iceweasel-*` packages select some useful Firefox extensions that 580 | happen to be packaged for Debian. We will look at other ways to 581 | customize extensions later. 582 | 583 | ##### Adding custom packages (including the guest kernel) 584 | 585 | For packages in the form of `.deb` files that are not part of your 586 | chosen Debian distribution (or that you want to override those in the 587 | distribution), simply place the files in the `config/packages.chroot` 588 | directory. This is how we provide the guest kernel package we built 589 | earlier. For example: 590 | 591 | ``` 592 | $ ls config/packages.chroot 593 | firejail_0.9.22_1_amd64.deb 594 | linux-image-3.19.1-grsec1_3.19.1-grsec1-1_amd64.deb 595 | live-boot_5.0~a1-1_all.deb 596 | live-boot-initramfs-tools_5.0~a1-1_all.deb 597 | live-config_5.0~a1-1_all.deb 598 | live-config-systemd_5.0~a1-1_all.deb 599 | ``` 600 | 601 | - A [Firejail](https://l3net.wordpress.com/projects/firejail/) package 602 | is included since it is not yet part of Debian. 603 | - The `linux-image` package is the guest kernel built earlier. 604 | - The `live-boot` packages are included here if they were custom-built 605 | for OverlayFS support as discussed earlier. 606 | - The `live-config` packages are included here from `experimental` so 607 | that they match the `live-boot` package versions. 608 | 609 | #### Enabling OverlayFS 610 | 611 | In order to use OverlayFS, you must edit the file `config/chroot` and 612 | change 613 | 614 | ``` 615 | LB_UNION_FILESYSTEM="aufs" 616 | ``` 617 | 618 | to 619 | 620 | ``` 621 | LB_UNION_FILESYSTEM="overlay" 622 | ``` 623 | 624 | You must also pass a boot parameter to the kernel; see below. 625 | 626 | #### Customizing Kernel Boot Parameters 627 | 628 | In some cases it is necessary to modify the boot parameters passed to 629 | the live image kernel. This is done by modifying the 630 | `LB_BOOTAPPEND_LIVE` option in `config/binary`, which defaults to: 631 | 632 | ``` 633 | LB_BOOTAPPEND_LIVE="boot=live components quiet splash" 634 | ``` 635 | 636 | Adding `union=overlay` is **required** for OverlayFS to work: 637 | 638 | ``` 639 | LB_BOOTAPPEND_LIVE="boot=live components quiet splash union=overlay" 640 | ``` 641 | 642 | Many runtime live image configuration options can also be passed in the 643 | form of kernel options; see 644 | [man live-config](http://live-systems.org/manpages/4.x/en/html/live-config.7.html) 645 | for details. 646 | 647 | #### Custom Build Hooks 648 | 649 | Executable scripts can be placed in the `config/hooks` directory. Their 650 | filenames should end with `.hook.chroot`. These scripts will be run 651 | inside the chroot at the end of the build process, and can be used to 652 | make changes to the live system before it is frozen into the final 653 | image. 654 | 655 | For example, this mechanism can be used to set a root password. Create 656 | an executable file called `config/hooks/0500-root-password.hook.chroot` 657 | containing: 658 | 659 | ```shell 660 | #!/bin/sh 661 | usermod -p '$5$GxpbCxvGOmtudd$uL90C.ZOMY6WxI4.x32kTCv38dGiXYUlfWGzCQuHmr3' root 662 | ``` 663 | 664 | Replace the argument to the `-p` option with the output of `mkpasswd` 665 | (available as part of the [whois](https://packages.debian.org/whois) 666 | package). See `man mkpasswd`. 667 | 668 | #### Customizing Live Image Contents 669 | 670 | The directory `config/includes.chroot` represents the `/` directory of 671 | the live image filesystem. Files placed here will be added directly to 672 | the final live image, replacing existing ones if necessary. 673 | 674 | For example, to add some custom 'dotfiles' to the live user's home 675 | directory, put them in `config/includes.chroot/etc/skel`. They will 676 | then end up in `/etc/skel` on the live system and will be copied to the 677 | live user's home directory when it is created during system startup. 678 | 679 | #### Custom Runtime Hooks 680 | 681 | Whereas build hooks are run during the live image build process, runtime 682 | hooks are run inside the live system during boot. These hooks are 683 | processed by `live-config`. To add a runtime hook, place an executable 684 | script in `config/includes.chroot/lib/live/config`. You should examine 685 | the default hooks (found in `chroot/lib/live/config` after an `lb 686 | build`) or the live-config examples in 687 | `/usr/share/doc/live-config/examples/hooks`. 688 | 689 | ### Step 2.4: Specific Customizations 690 | 691 | #### Root Access 692 | 693 | By default, the live system has one user named `user` with password 694 | `live`. This user can execute commands as root with `sudo` without 695 | specifying a password. A better configuration is to disable sudo and 696 | set a root password. We saw how to set a root password with a 697 | [Custom Build Hook](#custom-build-hooks) above. To disable sudo, add 698 | `live-config.noroot` to the kernel boot parameters list 699 | (`LB_BOOTAPPEND_LIVE` in `config/binary`). 700 | 701 | #### PaX Flags and Iceweasel 702 | 703 | If your guest kernel is configured for PaX and the process memory 704 | protection feature (`CONFIG_PAX_MPROTECT`) is enabled by default 705 | (recommended), then you will not be able to run Iceweasel without 706 | disabling memory protection on the `iceweasel` and `plugin-container` 707 | binaries in `/usr/lib/iceweasel`. (This suboptimal situation arises 708 | because Firefox uses JIT compilation for performance, which by its 709 | nature depends on memory regions that are both writable and executable. 710 | It is possible to compile Firefox without JIT, but this is not done for 711 | the standard Debian packages.) This can be done with a 712 | [runtime hook](#custom-runtime-hooks). For example, you can place an 713 | executable script like the following in 714 | `config/includes.chroot/lib/live/config/1200-setattr`: 715 | 716 | ```shell 717 | #!/bin/sh 718 | 719 | Init () 720 | { 721 | echo -n " setattr" 722 | } 723 | 724 | Config () 725 | { 726 | if [ -f /usr/bin/iceweasel -a -f /usr/bin/setfattr ]; then 727 | setfattr -n user.pax.flags -v m /usr/bin/iceweasel 728 | setfattr -n user.pax.flags -v m /usr/lib/iceweasel/plugin-container 729 | fi 730 | 731 | # Creating state file 732 | touch /var/lib/live/config/setattr 733 | } 734 | 735 | Init 736 | Config 737 | ``` 738 | 739 | The `setfattr` method only applies if you selected the 740 | `CONFIG_PAX_XATTR_PAX_FLAGS` option when configuring the guest kernel 741 | (recommended). 742 | 743 | #### Default X Resolution 744 | 745 | The X resolution can be changed at runtime with `xrandr`. To change the 746 | default resolution, add a 747 | [kernel boot parameter](#customizing-kernel-boot-parameters) like 748 | `live-config.xorg-resolution=1600x1200`. See 749 | [man live-config](http://live-systems.org/manpages/4.x/en/html/live-config.7.html) 750 | for details and other options. 751 | 752 | #### Mount a Host Directory with [9p](https://www.kernel.org/doc/Documentation/filesystems/9p.txt) 753 | 754 | If you configured QEMU to share a host directory via 9p and you included 755 | 9p support in your guest kernel, you may want to mount the share 756 | automatically at boot. There are many ways to do this, but a quick hack 757 | that works is to create `config/includes.chroot/etc/rc.local` as an 758 | executable shell script and add a line like: 759 | 760 | ``` 761 | mount -t 9p -o trans=virtio,version=9p2000.L share /mnt 762 | ``` 763 | 764 | Here `share` is a tag that must match the tag you passed to QEMU via the 765 | `-virtfs` option. 766 | 767 | Be warned that 9p support can exhibit some instabilities. In testing 768 | with Linux 3.18 and QEMU 2.1, passing `-sandbox on` caused writes to 769 | files over 9p to hang the writing process unrecoverably. Even without 770 | `-sandbox on`, creation of files over 9p fails with 'Operation not 771 | supported' (but writing to existing files works). This can be worked 772 | around by creating the file on the host before trying to write to it 773 | from the guest. 774 | 775 | Unrecoverable hangs on the guest may also arise with 9p following 776 | suspend/resume. To work around this, unload the 9p kernel modules 777 | (`9p`, `9pnet`, `9pnet_virtio`) using `modprobe -r` before suspend. 778 | 779 | ## Layer 3: Firejail 780 | 781 | [Firejail](https://l3net.wordpress.com/projects/firejail/) can 782 | optionally be used to provide another sandbox layer inside the live 783 | system. Firejail runs a program inside a 'container' environment that 784 | provides kernel namespace and filesystem isolation, as well as 785 | [seccomp](https://wiki.mozilla.org/Security/Sandbox/Seccomp) system call 786 | restriction. It is easy to use, supports a number of useful features 787 | and options, and can be used to sandbox any program. 788 | 789 | To install Firejail, simply download the `.deb` file from the Firejail 790 | project page (it is not yet available in the Debian repositories) and 791 | place it in `config/packages.chroot`. 792 | 793 | To run Firefox/Iceweasel with Firejail, run a command like: 794 | 795 | ``` 796 | $ firejail --debug iceweasel 797 | ``` 798 | 799 | The `--debug` option causes Firejail to produce verbose messages about 800 | what it's doing. Another useful mode is: 801 | 802 | ``` 803 | $ mkdir ~/sandbox 804 | $ firejail --debug --private=$HOME/sandbox iceweasel 805 | ``` 806 | 807 | The `--private` option starts the program in a clean home directory, 808 | preventing access to the user's real home directory. This is a good way 809 | to start a 'known clean' browser instance. 810 | 811 | ### Restricting Filesystem Access 812 | 813 | Firejail uses a blacklist method for restricting access to parts of 814 | filesystem. It achieves this by remounting system directories like 815 | `/etc`, `/lib`, and `/usr` as read-only within the container by default, 816 | and fully blocking access to some files and directories by mounting an 817 | empty tmpfs filesystem on top of them. (Seccomp filtering prevents 818 | these mounts from being changed within the container with calls to 819 | mount(2).) 820 | 821 | One omission in the defaults is that they leave the `/dev` directory 822 | accessible. This can be fixed by, for example, creating a file called 823 | `/etc/firejail/blacklist-dev.inc` with contents like: 824 | ``` 825 | seccomp mknod 826 | 827 | blacklist /dev/autofs 828 | blacklist /dev/block 829 | ... 830 | ``` 831 | 832 | The `seccomp mknod` line instructs Firejail to prevent the use of 833 | mknod(2) within the container so that new device files can't be created 834 | (the default in recent versions). To prevent access to the pre-existing 835 | device files in `/dev`, each file or subdirectory of `/dev` must be 836 | listed on a separate `blacklist` line. It is safe (and recommended) to 837 | blacklist nearly all device files (other than e.g. `/dev/null` and 838 | `/dev/urandom`) when running Firefox/Iceweasel. 839 | 840 | To use this file, add a line like 841 | 842 | ``` 843 | include /etc/firejail/blacklist-dev.inc 844 | ``` 845 | 846 | to `/etc/firejail/firefox.profile` or any other profile in `/etc/firejail`. 847 | 848 | ## Layer 4: Firefox/Iceweasel Customization 849 | 850 | If you are building a Firefox live image, you will want to customize 851 | things like the browser version, preferences, and extensions. 852 | 853 | ### Customizing the Browser Version 854 | 855 | If you want to install a version of Iceweasel other than the one in your 856 | target live image distribution, you will most likely have to resort to 857 | [APT pinning](https://wiki.debian.org/AptPreferences). For instance, 858 | you may want to install the latest Firefox release version, which is 859 | usually available in Debian `experimental`. You can do this by placing 860 | a couple of files in the `config/archives` directory. 861 | 862 | The `experimental.list.chroot` file: 863 | 864 | ``` 865 | deb http://ftp.debian.org/debian unstable main 866 | deb http://ftp.debian.org/debian experimental main 867 | ``` 868 | 869 | The `experimental.pref.chroot` file: 870 | 871 | ``` 872 | Package: iceweasel 873 | Pin: release a=experimental 874 | Pin-Priority: 995 875 | 876 | Package: * 877 | Pin: release a=unstable 878 | Pin-Priority: 1 879 | ``` 880 | 881 | This configuration directs APT to install the `iceweasel` package from 882 | `experimental` and to meet dependencies from `unstable` when they cannot 883 | be met from the target distribution. 884 | 885 | ### Customizing Preferences 886 | 887 | Most preferences are customized by placing files in 888 | `config/includes.chroot/etc/iceweasel/profile`. When a user first 889 | launches Iceweasel, it creates a profile directory in 890 | `~/.mozilla/firefox` based on the contents of `/etc/iceweasel/profile`. 891 | 892 | Some files and directories that are useful for customization: 893 | 894 | - `prefs.js` is the main browser settings and preferences file. 895 | - `bookmarks.html` is the default bookmarks list. 896 | - `extension-data` is a directory that some extensions use to store 897 | local data. 898 | - `searchplugins` is the directory where different search engines are 899 | defined. Each has a single `.xml` file in this directory. 900 | - `search-metadata.json` contains user preferences related to search 901 | engines. 902 | 903 | You can easily customize your live image browser by copying files like 904 | those above from the profile directory of an already-customized Firefox 905 | installation into `config/includes.chroot/etc/iceweasel/profile`. 906 | 907 | ### Default Preferences for Security and Privacy 908 | 909 | Many of the default Firefox settings are suboptimal for security and 910 | privacy. By way of example, this section provides a better set of 911 | defaults that can be included in your `prefs.js` file. You should 912 | understand what each setting does before using it — see the 913 | Mozilla [documentation](http://kb.mozillazine.org/About:config_entries) 914 | for details. 915 | 916 | ```javascript 917 | /* Don't start finding text when any input is typed */ 918 | user_pref("accessibility.typeaheadfind.autostart", false); 919 | 920 | /* Start with a simple blank page */ 921 | user_pref("browser.startup.page", 1); 922 | user_pref("browser.startup.homepage", "about:about"); 923 | 924 | /* Don't send every URL we visit to Google */ 925 | user_pref("browser.safebrowsing.enabled", false); 926 | user_pref("browser.safebrowsing.malware.enabled", false); 927 | 928 | /* Don't save information entered in web forms and search bars */ 929 | user_pref("browser.formfill.enable", false); 930 | 931 | /* Backspace goes back one page in history */ 932 | user_pref("browser.backspace_action", 0); 933 | 934 | /* Ask where to save downloads */ 935 | user_pref("browser.download.useDownloadDir", false); 936 | 937 | /* When JavaScript wants to open a new window, open a tab instead */ 938 | user_pref("browser.link.open_newwindow.restriction", 0); 939 | 940 | /* Always use Private Browsing Mode */ 941 | user_pref("browser.privatebrowsing.autostart", true); 942 | 943 | /* Disable search suggestions */ 944 | user_pref("browser.search.suggest.enabled", false); 945 | 946 | /* Limit state information saved between sessions */ 947 | user_pref("browser.sessionstore.max_tabs_undo", 5); 948 | user_pref("browser.sessionstore.resume_from_crash", false); 949 | 950 | /* Turn off tab animations */ 951 | user_pref("browser.tabs.animate", false); 952 | 953 | /* Turn off URL bar funny business */ 954 | user_pref("browser.urlbar.autocomplete.enabled", false); 955 | user_pref("browser.urlbar.trimURLs", false); 956 | 957 | /* Disallow JavaScript access to potentially dangerous APIs */ 958 | user_pref("dom.event.clipboardevents.enabled", false); 959 | user_pref("dom.battery.enabled", false); 960 | user_pref("dom.disable_window_open_feature.menubar", true); 961 | user_pref("dom.disable_window_open_feature.personalbar", true); 962 | user_pref("dom.disable_window_open_feature.scrollbars", true); 963 | user_pref("dom.disable_window_open_feature.toolbar", true); 964 | user_pref("dom.popup_maximum", 10); 965 | user_pref("dom.storage.default_quota", 0); 966 | 967 | /* Override User-Agent data to mitigate browser fingerprinting. 968 | * See https://panopticlick.eff.org/ 969 | */ 970 | user_pref("general.appname.override", "Netscape"); 971 | user_pref("general.appversion.override", "5.0 (Windows)"); 972 | user_pref("general.buildID.override", 0); 973 | user_pref("general.oscpu.override", "Windows NT 6.2"); 974 | user_pref("general.platform.override", "Win32"); 975 | user_pref("general.productSub.override", "20100101"); 976 | user_pref("general.useragent.override", "Mozilla/5.0 (Windows NT 6.2; rv:36.0) Gecko/20100101 Firefox/36.0"); 977 | user_pref("general.useragent.vendor", ""); 978 | user_pref("general.useragent.vendorSub", ""); 979 | user_pref("general.warnOnAboutConfig", false); 980 | user_pref("intl.accept_languages", "en-us,en;q=0.5"); 981 | 982 | /* Turn off "location aware browsing" */ 983 | user_pref("geo.enabled", false); 984 | 985 | /* Turn off sending non-URL words entered in the URL bar to Google */ 986 | user_pref("keyword.enabled", false); 987 | 988 | /* Turn off spell-checking */ 989 | user_pref("layout.spellcheckDefault", 0); 990 | 991 | /* Disable cookies by default. Use an extension for site-specific 992 | * whitelisting. 993 | */ 994 | user_pref("network.cookie.cookieBehavior", 2); 995 | user_pref("network.cookie.thirdparty.sessionOnly", true); 996 | 997 | /* Disable IPv6 unless you want to use it. */ 998 | user_pref("network.dns.disableIPv6", true); 999 | 1000 | /* Don't "proactively" perform DNS resolution */ 1001 | user_pref("network.dns.disablePrefetch", true); 1002 | 1003 | /* Unneeded unless we're using v6 */ 1004 | user_pref("network.http.fast-fallback-to-IPv4", false); 1005 | 1006 | /* Enable pipelining for better performance */ 1007 | user_pref("network.http.pipelining", true); 1008 | user_pref("network.http.pipelining.maxrequests", 15); 1009 | user_pref("network.http.pipelining.ssl", true); 1010 | user_pref("network.http.proxy.pipelining", true); 1011 | user_pref("network.http.redirection-limit", 5); 1012 | 1013 | /* Don't "proactively" fetch pages that haven't been requested */ 1014 | user_pref("network.prefetch-next", false); 1015 | 1016 | /* Disable Websockets by default */ 1017 | user_pref("network.websocket.enabled", false); 1018 | 1019 | /* Enable the Do-Not-Track header in HTTP requests */ 1020 | user_pref("privacy.donottrackheader.enabled", true); 1021 | 1022 | /* Clear Private Data when closing the browser */ 1023 | user_pref("privacy.sanitize.sanitizeOnShutdown", true); 1024 | 1025 | /* Disable unsafe RC4 ciphers */ 1026 | user_pref("security.ssl3.ecdhe_ecdsa_rc4_128_sha", false); 1027 | user_pref("security.ssl3.ecdhe_rsa_rc4_128_sha", false); 1028 | user_pref("security.ssl3.rsa_rc4_128_md5", false); 1029 | user_pref("security.ssl3.rsa_rc4_128_sha", false); 1030 | 1031 | /* Disable WebGL by default */ 1032 | user_pref("webgl.disabled", true); 1033 | ``` 1034 | 1035 | ### Customizing Extensions 1036 | 1037 | If the extension you want to add to your live image is already packaged 1038 | for Debian (for example, `iceweasel-noscript`), you can install it as 1039 | you would any other package by including its name in a file in 1040 | `config/package-lists`. 1041 | 1042 | Otherwise, you will have to place either the packed or unpacked 1043 | extension in the following directory: 1044 | 1045 | ``` 1046 | config/includes.chroot/usr/share/mozilla/extensions/{ec8030f7-c20a-464f-9b0e-13a3a9e97384} 1047 | ``` 1048 | 1049 | However, the packed extension (`.xpi`) file or unpacked directory must 1050 | have a specific name. To find this name, unpack the `install.rdf` file 1051 | and inspect it: 1052 | 1053 | ``` 1054 | $ unzip extension.xpi install.rdf 1055 | ``` 1056 | 1057 | Near the top of this file there will be an `` or `` tag that 1058 | looks similar to 1059 | 1060 | ``` 1061 | {2b10c1c8-a11f-4bad-fe9c-1c11e82cac42} 1062 | ``` 1063 | 1064 | or 1065 | 1066 | ``` 1067 | https-everywhere@eff.org 1068 | ``` 1069 | 1070 | To install an extension as a packed file, place it in the above 1071 | directory with the name `${XID}.xpi`, where `${XID}` is the content of 1072 | the `` or `` tag in the extension's `install.rdf` file. 1073 | 1074 | To install an extension as an unpacked directory, create a subdirectory 1075 | in the above directory whose name is `${XID}`, then place the unzipped 1076 | contents of the `.xpi` file in that directory. 1077 | 1078 | --------------------------------------------------------------------------------