web-archive/AlpineLinux/Alpine Linux on Raspberry Pi Diskless Mode with persistent storage.html
lauralani 46da301064
All checks were successful
ci/woodpecker/push/upload Pipeline was successful
initial commit
2023-09-01 08:20:19 +02:00

73 lines
No EOL
122 KiB
HTML
Raw Permalink Blame History

This file contains ambiguous Unicode characters

This file contains Unicode characters that might be confused with other characters. If you think that this is intentional, you can safely ignore this warning. Use the Escape button to reveal them.

<!DOCTYPE html> <html dir=auto style lang=en><!--
Page saved with SingleFile
url: https://thiagowfx.github.io/2022/01/alpine-linux-on-raspberry-pi-diskless-mode-with-persistent-storage/
saved date: Thu Mar 09 2023 10:38:11 GMT+0100 (Central European Standard Time)
--><meta charset=utf-8><meta http-equiv=x-ua-compatible content="IE=edge"><meta name=viewport content="width=device-width,initial-scale=1,shrink-to-fit=no"><meta name=robots content="index, follow"><title>★ Alpine Linux on Raspberry Pi: Diskless Mode with persistent storage | Not Just Serendipity</title><meta name=keywords content=linux,selfhosted><meta name=description content="Use case: Given an Alpine Linux diskless1 installation meant for
a Raspberry Pi setup, we would like to add a persistent storage component to it
to make it survive across reboots."><meta name=author content="Thiago Perrotta"><link rel=canonical href=https://thiagowfx.github.io/2022/01/alpine-linux-on-raspberry-pi-diskless-mode-with-persistent-storage/><style>:root{--gap:24px;--content-gap:20px;--nav-width:1024px;--main-width:720px;--header-height:60px;--footer-height:60px;--radius:8px;--theme:rgb(255,255,255);--entry:rgb(255,255,255);--primary:rgb(30,30,30);--secondary:rgb(108,108,108);--tertiary:rgb(214,214,214);--content:rgb(31,31,31);--hljs-bg:rgb(28,29,33);--code-bg:rgb(245,245,245);--border:rgb(238,238,238)}.dark{--theme:rgb(29,30,32);--entry:rgb(46,46,51);--primary:rgb(218,218,219);--secondary:rgb(155,156,157);--tertiary:rgb(65,66,68);--content:rgb(196,196,197);--hljs-bg:rgb(46,46,51);--code-bg:rgb(55,56,62);--border:rgb(51,51,51)}*,::after,::before{box-sizing:border-box}html{-webkit-tap-highlight-color:transparent;overflow-y:scroll}a,button,body,h1,h2,h3{color:var(--primary)}body{font-size:18px;line-height:1.6;word-break:break-word;background:var(--theme)}article,footer,header,main{display:block}h1,h2,h3{line-height:1.2}h1,h2,h3,p{margin-top:0;margin-bottom:0}ul{padding:0}a{text-decoration:none}body,ul{margin:0}button{padding:0;font:inherit;background:0 0;border:0}button{cursor:pointer}img{max-width:100%}.footer,.top-link{font-size:12px;color:var(--secondary)}.footer{max-width:calc(var(--main-width) + var(--gap)*2);margin:auto;padding:calc((var(--footer-height) - var(--gap))/2) var(--gap);text-align:center;line-height:24px}.footer span{margin-inline-start:1px;margin-inline-end:1px}.footer span:last-child{white-space:nowrap}.footer a{color:inherit;border-bottom:1px solid var(--secondary)}.footer a:hover{border-bottom:1px solid var(--primary)}.top-link{position:fixed;bottom:60px;right:30px;z-index:99;background:var(--tertiary);width:42px;height:42px;padding:12px;border-radius:64px;transition:visibility .5s,opacity .8s linear}.top-link,.top-link svg{filter:drop-shadow(0 0 0 var(--theme))}.footer a:hover,.top-link:hover{color:var(--primary)}.top-link:focus,#theme-toggle:focus{outline:0}.nav{display:flex;flex-wrap:wrap;justify-content:space-between;max-width:calc(var(--nav-width) + var(--gap)*2);margin-inline-start:auto;margin-inline-end:auto;line-height:var(--header-height)}.nav a{display:block}.logo,#menu{display:flex;margin:auto var(--gap)}.logo{flex-wrap:inherit}.logo a{font-size:24px;font-weight:700}.logo a img{display:inline;vertical-align:middle;pointer-events:none;transform:translate(0,-10%);border-radius:6px;margin-inline-end:8px}button#theme-toggle{font-size:26px;margin:auto 4px}body.dark #moon{vertical-align:middle;display:none}body:not(.dark) #sun{display:none}#menu{list-style:none;word-break:keep-all;overflow-x:auto;white-space:nowrap}#menu li+li{margin-inline-start:var(--gap)}#menu a{font-size:16px}.logo-switches{display:inline-flex;margin:auto 4px}.logo-switches{flex-wrap:inherit}.main{position:relative;min-height:calc(100vh - var(--header-height) - var(--footer-height));max-width:calc(var(--main-width) + var(--gap)*2);margin:auto;padding:var(--gap)}code{direction:ltr}div.highlight,pre{position:relative}.post-header{margin:24px auto var(--content-gap)}.post-title{margin-bottom:2px;font-size:40px}.post-meta{color:var(--secondary);font-size:14px;display:flex;flex-wrap:wrap}.post-content{color:var(--content)}.post-content h3{margin:24px 0 16px}.post-content h2{margin:32px auto 24px;font-size:32px}.post-content h3{font-size:24px}.post-content a,.toc a:hover{box-shadow:0 1px;box-decoration-break:clone;-webkit-box-decoration-break:clone}.post-content ol,.post-content p,.post-content ul{margin-bottom:var(--content-gap)}.post-content ol,.post-content ul{padding-inline-start:20px}.post-content li{margin-top:5px}.post-content li p{margin-bottom:0}.post-content .highlight:not(table){margin:10px auto;background:var(--hljs-bg)!important;border-radius:var(--radius);direction:ltr}.post-content .highlight pre{margin:0}.post-content code{margin:auto 4px;padding:4px 6px;font-size:.78em;line-height:1.5;background:var(--code-bg);border-radius:2px}.post-content pre code{display:block;margin:auto 0;padding:10px;color:#d5d5d6;background:var(--hljs-bg)!important;border-radius:var(--radius);overflow-x:auto;word-break:break-all}.post-content blockquote{margin:20px 0;padding:0 14px;border-inline-start:3px solid var(--primary)}.post-content hr{margin:30px 0;height:2px;background:var(--tertiary);border:0}.toc{margin:0 2px 40px;border:1px solid var(--border);border-radius:var(--radius);padding:.4em}.dark .toc{background:var(--entry)}.toc details summary{cursor:zoom-in;margin-inline-start:20px}.toc .details{display:inline;font-weight:500}.toc .inner{margin:0 20px;padding:10px 20px}.toc li ul{margin-inline-start:var(--gap)}.toc summary:focus{outline:0}.post-footer{margin-top:56px}.post-tags li{display:inline-block;margin-inline-end:3px;margin-bottom:5px}.post-tags a{border-radius:var(--radius);border:1px solid var(--border)}.post-tags a{display:block;padding-inline-start:14px;padding-inline-end:14px;color:var(--secondary);font-size:14px;line-height:34px;background:var(--code-bg)}.post-tags a:hover,.paginav a:hover{background:var(--border)}.hljs-variable{color:#eb3c54}.hljs-built_in,.hljs-meta{color:#e7ce56}.hljs-string{color:#4fb4d7}.hljs-keyword{color:#b45ea4}::-webkit-scrollbar-track{background:0 0}::-webkit-scrollbar-thumb{background:var(--tertiary);border:5px solid var(--theme);border-radius:var(--radius)}.post-content :not(table) ::-webkit-scrollbar-thumb{border:2px solid var(--hljs-bg);background:#717175}@media screen and (min-width:768px){::-webkit-scrollbar{width:19px;height:11px}}@media screen and (max-width:768px){:root{--gap:14px}.footer{padding:calc((var(--footer-height) - var(--gap) - 10px)/2) var(--gap)}}@media (prefers-reduced-motion){.top-link{transform:none}}</style>
<link rel=icon type=image/png sizes=32x32 href=""><meta name=theme-color content=#2e2e33><meta name=msapplication-TileColor content=#2e2e33><noscript><style>#theme-toggle,.top-link{display:none}</style><style>@media(prefers-color-scheme:dark){:root{--theme:rgb(29, 30, 32);--entry:rgb(46, 46, 51);--primary:rgb(218, 218, 219);--secondary:rgb(155, 156, 157);--tertiary:rgb(65, 66, 68);--content:rgb(196, 196, 197);--hljs-bg:rgb(46, 46, 51);--code-bg:rgb(55, 56, 62);--border:rgb(51, 51, 51)}.list{background:var(--theme)}.list:not(.dark)::-webkit-scrollbar-track{background:0 0}.list:not(.dark)::-webkit-scrollbar-thumb{border-color:var(--theme)}}</style></noscript><style>body{font-family:Crimson Pro,Vollkorn,Alegreya,Iowan Old Style,Apple Garamond,Baskerville,Times New Roman,Noto Serif,Droid Serif,Times,Source Serif Pro,serif,Apple Color Emoji,Segoe UI Emoji,Segoe UI Symbol,Noto Color Emoji}h1,h2,h3,footer,nav,.toc,.post-meta{font-family:Inter,Fira Sans,Lato,system-ui,-apple-system,BlinkMacSystemFont,Avenir Next,Avenir,Segoe UI,Helvetica Neue,Helvetica,Ubuntu,Roboto,Noto,Cantarell,Arial,sans-serif}code,pre{font-family:Fira Code,PT Mono,IBM Plex Mono,Menlo,Consolas,Monaco,Liberation Mono,Ubuntu Mono,Lucida Console,monospace}</style><meta property=og:title content="★ Alpine Linux on Raspberry Pi: Diskless Mode with persistent storage"><meta property=og:description content="Use case: Given an Alpine Linux diskless1 installation meant for
a Raspberry Pi setup, we would like to add a persistent storage component to it
to make it survive across reboots."><meta property=og:type content=article><meta property=og:url content=https://thiagowfx.github.io/2022/01/alpine-linux-on-raspberry-pi-diskless-mode-with-persistent-storage/><meta property=article:section content=posts><meta property=article:published_time content=2022-01-15T23:18:56-05:00><meta property=article:modified_time content=2022-01-15T23:18:56-05:00><meta name=twitter:card content=summary><meta name=twitter:title content="★ Alpine Linux on Raspberry Pi: Diskless Mode with persistent storage"><meta name=twitter:description content="Use case: Given an Alpine Linux diskless1 installation meant for
a Raspberry Pi setup, we would like to add a persistent storage component to it
to make it survive across reboots."><script type=application/ld+json>{"@context":"https://schema.org","@type":"BreadcrumbList","itemListElement":[{"@type":"ListItem","position":1,"name":"Posts","item":"https://thiagowfx.github.io/posts/"},{"@type":"ListItem","position":2,"name":"★ Alpine Linux on Raspberry Pi: Diskless Mode with persistent storage","item":"https://thiagowfx.github.io/2022/01/alpine-linux-on-raspberry-pi-diskless-mode-with-persistent-storage/"}]}</script><script type=application/ld+json>{"@context":"https://schema.org","@type":"BlogPosting","headline":"★ Alpine Linux on Raspberry Pi: Diskless Mode with persistent storage","name":"★ Alpine Linux on Raspberry Pi: Diskless Mode with persistent storage","description":"Use case: Given an Alpine Linux diskless1 installation meant for a Raspberry Pi setup, we would like to add a persistent storage component to it to make it survive across reboots.\n","keywords":["linux","selfhosted"],"articleBody":"Use case: Given an Alpine Linux diskless1 installation meant for a Raspberry Pi setup, we would like to add a persistent storage component to it to make it survive across reboots.\nGoal The Alpine Linux Wiki covers most of the installation process, hence I will only document the bits that were lacking and/or confusing therein.\nMy use case is the following:\nGiven a Raspberry Pi 3B with an old 4GiB SD Card as CF storage2, install Alpine Linux in diskless mode. Find a way to preserve modifications in /etc and /var, as well as any installed packages through its apk package manager.\nLets follow the steps outlined in the wiki.\nCopy Alpine to the SD Card Grab the SD card and install Alpine Linux in it.\nAlpine provides officially supported images designed for the Raspberry Pi.\nMost Linux distributions provide an .iso or .img file to be installed with a tool like Balena Etcher, Rufus, Raspberry Pi Imager or plain dd3.\nAlpine is not like most Linux distributions: Instead, it provides a .tar.gz archive with files that should be copied directly to the SD card. Grab the latest version (3.15 at the time of this post) from https://alpinelinux.org/downloads/. There are 3 options:\narmhf: Works with all Pis, but may perform less optimally on recent versions.\narmv7: Works with the Pi 3B, 32-bit.\naarch64: Works with the Pi 3B, 64-bit.\nI opted for aarch64 to make it 64-bit, but armv7 would also have worked well for my setup. In fact, Raspberry Pi OS (Debian) uses armv7 (32-bit) at the time of this writing.\nBefore copying files over, format the SD Card. As I was doing this from a Windows machine because it was the only one I had readily available with a SD card slot, I just used the native Windows Disk Management tool to do so. I decided to allocate a 100MB4 FAT32 partition. The rest of the SD card would be blank for now. Alpine is surprisingly small, 100MB was more than enough for the kernel and other needed files.\nOnce the SD card is formatted, copy the files over to it. It turns out Windows cannot extract tarballs (.tar.gz); a tool like 7-zip should do the job. Copy the files over to the root of the newly allocated FAT32 partition, and then safely eject the SD card.\nBoot Alpine from the SD Card The next step is to insert the SD Card into the Pi and then boot. I had some trouble in this step and eventually figured out I didnt mark the primary FAT32 partition as bootable. Unfortunately its not straightforward to mark the partition as bootable from Windows. On a Linux machine theres a wide array of tools to do so: fdisk, cfdisk (TUI), sfdisk (scriptable fdisk), parted, gparted (GUI) are some of them. I worked around that by installing Raspberry Pi OS on the SD card with the Raspberry Pi imager, and then overwriting it with the Alpine files. This works because the Raspberry PI OS installation marks the FAT32 partition as bootable.\nInstall Alpine Installing Alpine is well documented in the wiki thus it wont be covered here. It basically comes down to invoking setup-alpine, which then invokes other setup-* scripts.\nKeep in mind were not really “installing” Alpine as this is a diskless installation. A more accurate term here would be “configuring”.\nBefore invoking the installation script, I created a second primary partition in the SD card, set to ext4:\n# Configure networking to get working internet access. % setup-interfaces # Install some partitioning tools. % apk add cfdisk e2fsprogs # Create a second partition (mmcblk0p2) and write it. % cfdisk /dev/mmcblk0 # Format the partition as ext4. % mkfs.ext4 /dev/mmcblk0p2 # Mount the partition under /media. % mount /dev/mmcblk0p2 /media/mmcblk0p2 The installation is straightforward, we just need to pay attention to a few select steps:\nsetup-disk: Select none to ensure a diskless installation5. setup-apkcache: Select /media/mmcblk0p2/cache to persist downloaded apk packages. setup-lbu: Edit /etc/lbu/lbu.conf and set LBU_MEDIA=\"mmcblk0p2\". Note: Do not add /media as it is implicit. Once the installation is complete, run lbu commit to persist the changes in the second partition. Once you do so, a .apkovl.tar.gz6 file should materialize on /media/mmcblk0p2/.\nThis is a good moment to reboot. Before we do so, lets cache the packages we had previously downloaded.\n# Cache packages. % apk cache download % reboot After the first reboot If everything worked as expected, once you reboot all your previously installed packages should have been preserved and automatically restored / reinstalled, as well as your modifications done to /etc.\nFrom this point on, whenever you install a new package that you want to be preserved for subsequent reboots, run lbu commit afterwards. For example:\n% apk add vim % lbu commit If you would like to see what is going to be committed, run lbu status or lbu diff before doing the actual commit. Whenever you commit, /media/mmcblk0p2/.apkovl.tar.gz gets overwritten with your most recent modifications.\nIts possible to keep more than one backup file by changing BACKUP_LIMIT= in /etc/lbu/lbu.conf. This is specially handy if you decide to revert to an earlier system snapshot / state later on. The stock config looks like this:\n% cat /etc/lbu/lbu.conf # what cipher to use with -e option DEFAULT_CIPHER=aes-256-cbc # Uncomment the row below to encrypt config by default # ENCRYPTION=$DEFAULT_CIPHER # Uncomment below to avoid option to 'lbu commit' # Can also be set to 'floppy' # LBU_MEDIA=usb # Set the LBU_BACKUPDIR variable in case you prefer to save the apkovls # in a normal directory instead of mounting an external media. # LBU_BACKUPDIR=/root/config-backups # Uncomment below to let lbu make up to 3 backups # BACKUP_LIMIT=3 Tip: You can find the list of all explicitly installed packages in /etc/apk/world.\nThe last piece: make /var persistent There are three natural ways that come to mind to make /var persistent:\nA) Separate partition (or file) Instead of two partitions (FAT32 and ext4), create 3 partitions: FAT32, ext4 and ext4. Use the latter one to mount /var on, saving this information in /etc/fstab. The main disadvantage of this setup is that youll need to allocate a fixed amount of space of each of the ext4 partitions and it may be difficult to figure out how to split the space between them.\nA variant of this approach is to just create the third partition as a file:\n# 500MB file % dd if=/dev/zero of=/media/mmcblk0p2/var.img bs=1M count=500 status=progress % mkfs.ext4 /media/mmcblk0p2/var.img % mount /media/mmcblk0p2/var.img /var This works because the Linux kernel supports mounting files as if they were device blocks, treating them as loop devices (pseudo-devices).\nI dont like these approaches because they shadow the preexisting /var from the boot media, which in turn messes up with existing services that use it such as cron: % crontab -l would fail. One workaround would be to mount a /var subdirectory instead: for example, /var/lib/docker for docker.\nB) Bind mount This one is straightforward:\n% mount --bind /media/mmcblk0p2/var/lib/docker /var/lib/docker The actual partition lives in the SD card, however we make a bind mount under /var, which is like an alias. From Stack Exchange:\nA bind mount is an alternate view of a directory tree. Classically, mounting creates a view of a storage device as a directory tree. A bind mount instead takes an existing directory tree and replicates it under a different point. The directories and files in the bind mount are the same as the original. Any modification on one side is immediately reflected on the other side, since the two views show the same data.\nC) Overlay mount From ArchWiki:\nOverlayfs allows one, usually read-write, directory tree to be overlaid onto another, read-only directory tree. All modifications go to the upper, writable layer. This type of mechanism is most often used for live CDs but there is a wide variety of other uses.\nIts perfect for our use case, which uses a live bootable SD card for Alpine. It blends the preexisting, ephemeral, in-memory /var with the persistent in-disk /var.\nI wanted to mount /var directly but found it to be problematic for the same reasons mentioned earlier, therefore I just went with /var/lib/docker instead:\n# Create overlay upper and work directories. % mkdir -p /media/mmcblk0p2/var/lib/docker /media/mmcblk0p2/var/lib/docker-work # Add mountpoint entry to fstab. Note: The work dir must be an empty directory in the same filesystem mount as the upper directory. % echo \"overlay /var/lib/docker overlay lowerdir=/var/lib/docker,upperdir=/media/mmcblk0p2/var/lib/docker,workdir=/media/mmcblk0p2/var/lib/docker-work 0 0\" >> /etc/fstab # Mount all fstab entries, including our newly added one. % mount -a Conclusion I opted for the third approach, using an overlay mount, it was the most seamless one. A bind mount would have been fine as well.\nThe final setup works surprisingly well:\nAlpine Linux is very lightweight and runs mostly from RAM apk cache is persistent to the ext4 partition /var/ is persistent to the ext4 partition lbu commit persists changes in /etc/ and /home/ in the ext4 partition Every reboot fully resets the system sans persistent components above References https://vincentserpoul.github.io/post/alpine-linux-rpi0/ http://dahl-jacobsen.dk/tips/blog/2021-04-08-docker-on-alpine-linux/ http://dahl-jacobsen.dk/tips/blog/2018-03-15-alpine-on-raspberry-pi/ Running (almost) fully from RAM. ↩︎\nCF = Compact disk. ↩︎\nOn Linux Id usually opt for dd, on Windows the Raspberry Pi Imager is a sensible choice. ↩︎\n100MB is overly conservative, but keep in mind I had a very small SD Card, with only 4GiB storage. 250MB or even 500MB should be a more sensible default if you have a bigger SD Card (e.g. 32GiB). ↩︎\nAn alternative is to select data disk mode, but it didnt work for me. ↩︎\novl is short for overlay. Not to be confused with vol for volume. ↩︎\n","wordCount":"1596","inLanguage":"en","datePublished":"2022-01-15T23:18:56-05:00","dateModified":"2022-01-15T23:18:56-05:00","author":{"@type":"Person","name":"Thiago Perrotta"},"mainEntityOfPage":{"@type":"WebPage","@id":"https://thiagowfx.github.io/2022/01/alpine-linux-on-raspberry-pi-diskless-mode-with-persistent-storage/"},"publisher":{"@type":"Person","name":"Not Just Serendipity","logo":{"@type":"ImageObject","url":"https://thiagowfx.github.io/favicon.ico"}}}</script><style>.sf-hidden{display:none!important}</style><meta http-equiv=content-security-policy content="default-src 'none'; font-src 'self' data:; img-src 'self' data:; style-src 'unsafe-inline'; media-src 'self' data:; script-src 'unsafe-inline' data:; object-src 'self' data:; frame-src 'self' data:;"><style>img[src="data:,"],source[src="data:,"]{display:none!important}</style><body id=top class=dark><header class=header><nav class=nav><div class=logo><a href=https://thiagowfx.github.io/ accesskey=h title="Not Just Serendipity (Alt + H)"><img src= alt aria-label=logo height=32>Not Just Serendipity</a><div class=logo-switches><button id=theme-toggle accesskey=t title="(Alt + T)"><svg id=moon xmlns=http://www.w3.org/2000/svg width=24 height=18 viewBox="0 0 24 24" fill=none stroke=currentcolor stroke-width=2 stroke-linecap=round stroke-linejoin=round><path d="M21 12.79A9 9 0 1111.21 3 7 7 0 0021 12.79z"></path></svg><svg id=sun xmlns=http://www.w3.org/2000/svg width=24 height=18 viewBox="0 0 24 24" fill=none stroke=currentcolor stroke-width=2 stroke-linecap=round stroke-linejoin=round><circle cx=12 cy=12 r=5></circle><line x1=12 y1=1 x2=12 y2=3></line><line x1=12 y1=21 x2=12 y2=23></line><line x1=4.22 y1=4.22 x2=5.64 y2=5.64></line><line x1=18.36 y1=18.36 x2=19.78 y2=19.78></line><line x1=1 y1=12 x2=3 y2=12></line><line x1=21 y1=12 x2=23 y2=12></line><line x1=4.22 y1=19.78 x2=5.64 y2=18.36></line><line x1=18.36 y1=5.64 x2=19.78 y2=4.22></line></svg></button></div></div><ul id=menu><li><a href=https://thiagowfx.github.io/posts/ title=Blog><span>Blog</span></a><li><a href=https://thiagowfx.github.io/archives/ title=Archives><span>Archives</span></a><li><a href=https://thiagowfx.github.io/search/ title="Search (Alt + /)" accesskey=/><span>Search</span></a><li><a href=https://thiagowfx.github.io/tags/ title=Tags><span>Tags</span></a></ul></nav></header><main class=main><article class=post-single><header class=post-header><h1 class=post-title>★ Alpine Linux on Raspberry Pi: Diskless Mode with persistent storage</h1><div class=post-meta><span title="2022-01-15 23:18:56 -0500 -0500">January 15, 2022</span>&nbsp;·&nbsp;1596 words&nbsp;·&nbsp;Thiago Perrotta</div></header><div class=toc><details><template shadowroot=closed><slot name=internal-main-summary><summary>Details</summary></slot><slot></slot></template><summary accesskey=c title="(Alt + C)"><span class=details>Table of Contents</span></summary><div class=inner><ul><li><a href=#goal aria-label=Goal>Goal</a><li><a href=#copy-alpine-to-the-sd-card aria-label="Copy Alpine to the SD Card">Copy Alpine to the SD Card</a><li><a href=#boot-alpine-from-the-sd-card aria-label="Boot Alpine from the SD Card">Boot Alpine from the SD Card</a><li><a href=#install-alpine aria-label="Install Alpine">Install Alpine</a><li><a href=#after-the-first-reboot aria-label="After the first reboot">After the first reboot</a><li><a href=#the-last-piece-make-var-persistent aria-label="The last piece: make /var persistent">The last piece: make /var persistent</a><ul><li><a href=#a-separate-partition-or-file aria-label="A) Separate partition (or file)">A) Separate partition (or file)</a><li><a href=#b-bind-mount aria-label="B) Bind mount">B) Bind mount</a><li><a href=#c-overlay-mount aria-label="C) Overlay mount">C) Overlay mount</a></ul><li><a href=#conclusion aria-label=Conclusion>Conclusion</a><li><a href=#references aria-label=References>References</a></ul></div></details></div><div class=post-content><p>Use case: Given an Alpine Linux <strong>diskless</strong><sup id=fnref:1><a href=#fn:1 class=footnote-ref role=doc-noteref>1</a></sup> installation meant for
a Raspberry Pi setup, we would like to add a persistent storage component to it
to make it survive across reboots.<h2 id=goal>Goal<a class="anchor sf-hidden" aria-hidden=true href=#goal hidden>#</a></h2><p>The <a href=https://wiki.alpinelinux.org/wiki/Installation>Alpine Linux Wiki</a> covers most of the installation process, hence I will only document the bits that were lacking and/or confusing therein.<p>My use case is the following:<blockquote><p>Given a Raspberry Pi 3B with an old 4GiB SD Card as CF storage<sup id=fnref:2><a href=#fn:2 class=footnote-ref role=doc-noteref>2</a></sup>, install Alpine Linux in diskless mode. Find a way to preserve modifications in <code>/etc</code> and <code>/var</code>, as well as any installed packages through its <code>apk</code> package manager.</p></blockquote><p>Lets follow the steps outlined in the wiki.<h2 id=copy-alpine-to-the-sd-card>Copy Alpine to the SD Card<a class="anchor sf-hidden" aria-hidden=true href=#copy-alpine-to-the-sd-card hidden>#</a></h2><blockquote><p>Grab the SD card and install Alpine Linux in it.</p></blockquote><p>Alpine provides officially supported images designed for the Raspberry Pi.<p>Most Linux distributions provide an <code>.iso</code> or <code>.img</code> file to be installed with a tool like <a href=https://www.balena.io/etcher/>Balena Etcher</a>, <a href=https://rufus.ie/en/>Rufus</a>, <a href=https://www.raspberrypi.com/news/raspberry-pi-imager-imaging-utility/><strong>Raspberry Pi Imager</strong></a> or plain <code>dd</code><sup id=fnref:3><a href=#fn:3 class=footnote-ref role=doc-noteref>3</a></sup>.<p>Alpine is not like most Linux distributions: Instead, it provides a <code>.tar.gz</code> archive with files that should be copied directly to the SD card. Grab the latest version (3.15 at the time of this post) from <a href=https://alpinelinux.org/downloads/>https://alpinelinux.org/downloads/</a>. There are 3 options:<ul><li><p><code>armhf</code>: Works with all Pis, but may perform less optimally on recent versions.</p><li><p><code>armv7</code>: Works with the Pi 3B, 32-bit.</p><li><p><code>aarch64</code>: Works with the Pi 3B, 64-bit.</p></ul><p>I opted for <code>aarch64</code> to make it 64-bit, but <code>armv7</code> would also have worked well for my setup. In fact, Raspberry Pi OS (Debian) uses <code>armv7</code> (32-bit) at the time of this writing.<p>Before copying files over, format the SD Card. As I was doing this
from a Windows machine because it was the only one I had readily available with
a SD card slot, I just used the native Windows Disk Management tool to do so.
I decided to allocate a 100MB<sup id=fnref:4><a href=#fn:4 class=footnote-ref role=doc-noteref>4</a></sup> FAT32 partition. The rest of the SD card would be
blank for now. Alpine is surprisingly small, 100MB was more than enough for the kernel and other needed files.<p>Once the SD card is formatted, copy the files over to it. It turns out Windows cannot extract tarballs (<code>.tar.gz</code>); a tool like <a href=https://www.7-zip.org/>7-zip</a> should do the job. Copy the files over to the root of the newly allocated FAT32 partition, and then safely eject the SD card.<h2 id=boot-alpine-from-the-sd-card>Boot Alpine from the SD Card<a class="anchor sf-hidden" aria-hidden=true href=#boot-alpine-from-the-sd-card hidden>#</a></h2><p>The next step is to insert the SD Card into the Pi and then boot. I had some trouble in this step and eventually figured out I didnt mark the primary FAT32 partition as bootable. Unfortunately its not straightforward to mark the partition as bootable from Windows. On a Linux machine theres a wide array of tools to do so: <code>fdisk</code>, <code>cfdisk</code> (TUI), <code>sfdisk</code> (scriptable <code>fdisk</code>), <code>parted</code>, <code>gparted</code> (GUI) are some of them. I worked around that by installing Raspberry Pi OS on the SD card with the Raspberry Pi imager, and then overwriting it with the Alpine files. This works because the Raspberry PI OS installation marks the FAT32 partition as bootable.<h2 id=install-alpine>Install Alpine<a class="anchor sf-hidden" aria-hidden=true href=#install-alpine hidden>#</a></h2><p>Installing Alpine is well documented in the <a href=https://wiki.alpinelinux.org/wiki/Installation>wiki</a> thus it wont be covered here. It basically comes down to invoking <code>setup-alpine</code>, which then invokes other <code>setup-*</code> scripts.<p>Keep in mind were not really “installing” Alpine as this is a diskless installation. A more accurate term here would be “configuring”.<p>Before invoking the installation script, I created a second primary partition in the SD card, set to <code>ext4</code>:<div class=highlight><pre tabindex=0 style=color:#f8f8f2;background-color:#272822;-moz-tab-size:4;-o-tab-size:4;tab-size:4><code class="language-shell hljs" data-lang=shell><span style=display:flex><span><span style=color:#75715e><span class=hljs-meta>#</span><span class=bash> Configure networking to get working internet access.</span></span>
</span></span><span style=display:flex><span><span class=hljs-meta>%</span><span class=bash> setup-interfaces</span>
</span></span><span style=display:flex><span><span class=hljs-meta>
</span></span></span><span style=display:flex><span><span style=color:#75715e><span class=hljs-meta>#</span><span class=bash> Install some partitioning tools.</span></span>
</span></span><span style=display:flex><span><span class=hljs-meta>%</span><span class=bash> apk add cfdisk e2fsprogs</span>
</span></span><span style=display:flex><span><span class=hljs-meta>
</span></span></span><span style=display:flex><span><span style=color:#75715e><span class=hljs-meta>#</span><span class=bash> Create a second partition (mmcblk0p2) and write it.</span></span>
</span></span><span style=display:flex><span><span class=hljs-meta>%</span><span class=bash> cfdisk /dev/mmcblk0</span>
</span></span><span style=display:flex><span><span class=hljs-meta>
</span></span></span><span style=display:flex><span><span style=color:#75715e><span class=hljs-meta>#</span><span class=bash> Format the partition as ext4.</span></span>
</span></span><span style=display:flex><span><span class=hljs-meta>%</span><span class=bash> mkfs.ext4 /dev/mmcblk0p2</span>
</span></span><span style=display:flex><span><span class=hljs-meta>
</span></span></span><span style=display:flex><span><span style=color:#75715e><span class=hljs-meta>#</span><span class=bash> Mount the partition under /media.</span></span>
</span></span><span style=display:flex><span><span class=hljs-meta>%</span><span class=bash> mount /dev/mmcblk0p2 /media/mmcblk0p2</span>
</span></span></code></pre><button class="copy-code sf-hidden">copy</button></div><p>The installation is straightforward, we just need to pay attention to a few select steps:<ul><li><code>setup-disk</code>: Select <code>none</code> to ensure a <code>diskless</code> installation<sup id=fnref:5><a href=#fn:5 class=footnote-ref role=doc-noteref>5</a></sup>.<li><code>setup-apkcache</code>: Select <code>/media/mmcblk0p2/cache</code> to persist downloaded <code>apk</code> packages.<li><code>setup-lbu</code>: Edit <code>/etc/lbu/lbu.conf</code> and set <code>LBU_MEDIA="mmcblk0p2"</code>. Note: Do not add <code>/media</code> as it is implicit.</ul><p>Once the installation is complete, run <code>lbu commit</code> to persist the changes in the second partition. Once you do so, a <code>&lt;hostname&gt;.apkovl.tar.gz</code><sup id=fnref:6><a href=#fn:6 class=footnote-ref role=doc-noteref>6</a></sup> file should materialize on <code>/media/mmcblk0p2/</code>.<p>This is a good moment to reboot. Before we do so, lets cache the packages we had previously downloaded.<div class=highlight><pre tabindex=0 style=color:#f8f8f2;background-color:#272822;-moz-tab-size:4;-o-tab-size:4;tab-size:4><code class="language-shell hljs" data-lang=shell><span style=display:flex><span><span style=color:#75715e><span class=hljs-meta>#</span><span class=bash> Cache packages.</span></span>
</span></span><span style=display:flex><span><span class=hljs-meta>%</span><span class=bash> apk cache download</span>
</span></span><span style=display:flex><span><span class=hljs-meta>
</span></span></span><span style=display:flex><span><span class=hljs-meta>%</span><span class=bash> reboot</span>
</span></span></code></pre><button class="copy-code sf-hidden">copy</button></div><h2 id=after-the-first-reboot>After the first reboot<a class="anchor sf-hidden" aria-hidden=true href=#after-the-first-reboot hidden>#</a></h2><p>If everything worked as expected, once you reboot all your previously installed packages should have been preserved and automatically restored / reinstalled, as well as your modifications done to <code>/etc</code>.<p>From this point on, whenever you install a new package that you want to be preserved for subsequent reboots, run <code>lbu commit</code> afterwards. For example:<div class=highlight><pre tabindex=0 style=color:#f8f8f2;background-color:#272822;-moz-tab-size:4;-o-tab-size:4;tab-size:4><code class="language-shell hljs" data-lang=shell><span style=display:flex><span><span class=hljs-meta>%</span><span class=bash> apk add vim</span>
</span></span><span style=display:flex><span><span class=hljs-meta>%</span><span class=bash> lbu commit</span>
</span></span></code></pre><button class="copy-code sf-hidden">copy</button></div><p>If you would like to see what is going to be committed, run <code>lbu status</code> or <code>lbu diff</code> before doing the actual commit. Whenever you commit, <code>/media/mmcblk0p2/&lt;hostname&gt;.apkovl.tar.gz</code> gets overwritten with your most recent modifications.<p>Its possible to keep more than one backup file by changing <code>BACKUP_LIMIT=</code> in <code>/etc/lbu/lbu.conf</code>. This is specially handy if you decide to revert to an earlier system snapshot / state later on. The stock config looks like this:<div class=highlight><pre tabindex=0 style=color:#f8f8f2;background-color:#272822;-moz-tab-size:4;-o-tab-size:4;tab-size:4><code class="language-shell hljs" data-lang=shell><span style=display:flex><span><span class=hljs-meta>%</span><span class=bash> cat /etc/lbu/lbu.conf</span>
</span></span><span style=display:flex><span><span style=color:#75715e><span class=hljs-meta>#</span><span class=bash> what cipher to use with -e option</span></span>
</span></span><span style=display:flex><span>DEFAULT_CIPHER<span style=color:#f92672>=</span>aes-256-cbc
</span></span><span style=display:flex><span><span class=hljs-meta>
</span></span></span><span style=display:flex><span><span style=color:#75715e><span class=hljs-meta>#</span><span class=bash> Uncomment the row below to encrypt config by default</span></span>
</span></span><span style=display:flex><span><span style=color:#75715e><span class=hljs-meta>#</span><span class=bash> ENCRYPTION=<span class=hljs-variable>$DEFAULT_CIPHER</span></span></span>
</span></span><span style=display:flex><span><span class=hljs-meta>
</span></span></span><span style=display:flex><span><span style=color:#75715e><span class=hljs-meta>#</span><span class=bash> Uncomment below to avoid &lt;media&gt; option to <span class=hljs-string>'lbu commit'</span></span></span>
</span></span><span style=display:flex><span><span style=color:#75715e><span class=hljs-meta>#</span><span class=bash> Can also be <span class=hljs-built_in>set</span> to <span class=hljs-string>'floppy'</span></span></span>
</span></span><span style=display:flex><span><span style=color:#75715e><span class=hljs-meta>#</span><span class=bash> LBU_MEDIA=usb</span></span>
</span></span><span style=display:flex><span><span class=hljs-meta>
</span></span></span><span style=display:flex><span><span style=color:#75715e><span class=hljs-meta>#</span><span class=bash> Set the LBU_BACKUPDIR variable <span class=hljs-keyword>in</span> <span class=hljs-keyword>case</span> you prefer to save the apkovls</span></span>
</span></span><span style=display:flex><span><span style=color:#75715e><span class=hljs-meta>#</span><span class=bash> <span class=hljs-keyword>in</span> a normal directory instead of mounting an external media.</span></span>
</span></span><span style=display:flex><span><span style=color:#75715e><span class=hljs-meta>#</span><span class=bash> LBU_BACKUPDIR=/root/config-backups</span></span>
</span></span><span style=display:flex><span><span class=hljs-meta>
</span></span></span><span style=display:flex><span><span style=color:#75715e><span class=hljs-meta>#</span><span class=bash> Uncomment below to <span class=hljs-built_in>let</span> lbu make up to 3 backups</span></span>
</span></span><span style=display:flex><span><span style=color:#75715e><span class=hljs-meta>#</span><span class=bash> BACKUP_LIMIT=3</span></span>
</span></span></code></pre><button class="copy-code sf-hidden">copy</button></div><p><strong>Tip</strong>: You can find the list of all explicitly installed packages in <code>/etc/apk/world</code>.<h2 id=the-last-piece-make-var-persistent>The last piece: make /var persistent<a class="anchor sf-hidden" aria-hidden=true href=#the-last-piece-make-var-persistent hidden>#</a></h2><p>There are three natural ways that come to mind to make <code>/var</code> persistent:<h3 id=a-separate-partition-or-file>A) Separate partition (or file)<a class="anchor sf-hidden" aria-hidden=true href=#a-separate-partition-or-file hidden>#</a></h3><p>Instead of two partitions (FAT32 and ext4), create 3 partitions: FAT32, ext4 and ext4. Use the latter one to mount <code>/var</code> on, saving this information in <code>/etc/fstab</code>. The main disadvantage of this setup is that youll need to allocate a fixed amount of space of each of the ext4 partitions and it may be difficult to figure out how to split the space between them.<p>A variant of this approach is to just create the third partition as a file:<div class=highlight><pre tabindex=0 style=color:#f8f8f2;background-color:#272822;-moz-tab-size:4;-o-tab-size:4;tab-size:4><code class="language-shell hljs" data-lang=shell><span style=display:flex><span><span style=color:#75715e><span class=hljs-meta>#</span><span class=bash> 500MB file</span></span>
</span></span><span style=display:flex><span><span class=hljs-meta>%</span><span class=bash> dd </span><span style=color:#66d9ef><span class=bash><span class=hljs-keyword>if</span></span></span><span style=color:#f92672><span class=bash>=</span></span><span class=bash>/dev/zero of</span><span style=color:#f92672><span class=bash>=</span></span><span class=bash>/media/mmcblk0p2/var.img bs</span><span style=color:#f92672><span class=bash>=</span></span><span class=bash>1M count</span><span style=color:#f92672><span class=bash>=</span></span><span style=color:#ae81ff><span class=bash>500</span></span><span class=bash> status</span><span style=color:#f92672><span class=bash>=</span></span><span class=bash>progress</span>
</span></span><span style=display:flex><span><span class=hljs-meta>%</span><span class=bash> mkfs.ext4 /media/mmcblk0p2/var.img</span>
</span></span><span style=display:flex><span><span class=hljs-meta>%</span><span class=bash> mount /media/mmcblk0p2/var.img /var</span>
</span></span></code></pre><button class="copy-code sf-hidden">copy</button></div><p>This works because the Linux kernel supports mounting files as if they were device blocks, treating them as loop devices (pseudo-devices).<p>I dont like these approaches because they shadow the preexisting <code>/var</code> from the boot media, which in turn messes up with existing services that use it such as <code>cron</code>: <code>% crontab -l</code> would fail. One workaround would be to mount a <code>/var</code> subdirectory instead: for example, <code>/var/lib/docker</code> for docker.<h3 id=b-bind-mount>B) Bind mount<a class="anchor sf-hidden" aria-hidden=true href=#b-bind-mount hidden>#</a></h3><p>This one is straightforward:<div class=highlight><pre tabindex=0 style=color:#f8f8f2;background-color:#272822;-moz-tab-size:4;-o-tab-size:4;tab-size:4><code class="language-shell hljs" data-lang=shell><span style=display:flex><span><span class=hljs-meta>%</span><span class=bash> mount --<span class=hljs-built_in>bind</span> /media/mmcblk0p2/var/lib/docker /var/lib/docker</span>
</span></span></code></pre><button class="copy-code sf-hidden">copy</button></div><p>The actual partition lives in the SD card, however we make a bind mount under
<code>/var</code>, which is like an <em>alias</em>. From <a href=https://unix.stackexchange.com/questions/198590/what-is-a-bind-mount>Stack Exchange</a>:<blockquote><p>A bind mount is an alternate view of a directory tree. Classically, mounting creates a view of a storage device as a directory tree. A bind mount instead takes an existing directory tree and replicates it under a different point. The directories and files in the bind mount are the same as the original. Any modification on one side is immediately reflected on the other side, since the two views show the same data.</p></blockquote><h3 id=c-overlay-mount>C) Overlay mount<a class="anchor sf-hidden" aria-hidden=true href=#c-overlay-mount hidden>#</a></h3><p>From <a href=https://wiki.archlinux.org/title/Overlay_filesystem>ArchWiki</a>:<blockquote><p>Overlayfs allows one, usually read-write, directory tree to be overlaid onto another, read-only directory tree. All modifications go to the upper, writable layer. This type of mechanism is most often used for live CDs but there is a wide variety of other uses.</p></blockquote><p>Its perfect for our use case, which uses a live bootable SD card for Alpine. It blends the preexisting, ephemeral, in-memory <code>/var</code> with the persistent in-disk <code>/var</code>.<p>I wanted to mount <code>/var</code> directly but found it to be problematic for the same reasons mentioned earlier, therefore I just went with <code>/var/lib/docker</code> instead:<div class=highlight><pre tabindex=0 style=color:#f8f8f2;background-color:#272822;-moz-tab-size:4;-o-tab-size:4;tab-size:4><code class="language-shell hljs" data-lang=shell><span style=display:flex><span><span style=color:#75715e><span class=hljs-meta>#</span><span class=bash> Create overlay upper and work directories.</span></span>
</span></span><span style=display:flex><span><span class=hljs-meta>%</span><span class=bash> mkdir -p /media/mmcblk0p2/var/lib/docker /media/mmcblk0p2/var/lib/docker-work</span>
</span></span><span style=display:flex><span><span class=hljs-meta>
</span></span></span><span style=display:flex><span><span style=color:#75715e><span class=hljs-meta>#</span><span class=bash> Add mountpoint entry to fstab. Note: The work dir must be an empty directory <span class=hljs-keyword>in</span> the same filesystem mount as the upper directory.</span></span>
</span></span><span style=display:flex><span><span class=hljs-meta>%</span><span class=bash> <span class=hljs-built_in>echo</span> </span><span style=color:#e6db74><span class=bash><span class=hljs-string>"overlay /var/lib/docker overlay lowerdir=/var/lib/docker,upperdir=/media/mmcblk0p2/var/lib/docker,workdir=/media/mmcblk0p2/var/lib/docker-work 0 0"</span></span></span><span class=bash> &gt;&gt; /etc/fstab</span>
</span></span><span style=display:flex><span><span class=hljs-meta>
</span></span></span><span style=display:flex><span><span style=color:#75715e><span class=hljs-meta>#</span><span class=bash> Mount all fstab entries, including our newly added one.</span></span>
</span></span><span style=display:flex><span><span class=hljs-meta>%</span><span class=bash> mount -a</span>
</span></span></code></pre><button class="copy-code sf-hidden">copy</button></div><h2 id=conclusion>Conclusion<a class="anchor sf-hidden" aria-hidden=true href=#conclusion hidden>#</a></h2><p>I opted for the third approach, using an overlay mount, it was the most
seamless one. A bind mount would have been fine as well.<p>The final setup works surprisingly well:<ul><li>Alpine Linux is very lightweight and runs mostly from RAM<li><code>apk</code> cache is persistent to the ext4 partition<li><code>/var/</code> is persistent to the ext4 partition<li><code>lbu commit</code> persists changes in <code>/etc/</code> and <code>/home/</code> in the ext4 partition<li>Every reboot fully resets the system sans persistent components above</ul><h2 id=references>References<a class="anchor sf-hidden" aria-hidden=true href=#references hidden>#</a></h2><ul><li><a href=https://vincentserpoul.github.io/post/alpine-linux-rpi0/>https://vincentserpoul.github.io/post/alpine-linux-rpi0/</a><li><a href=http://dahl-jacobsen.dk/tips/blog/2021-04-08-docker-on-alpine-linux/>http://dahl-jacobsen.dk/tips/blog/2021-04-08-docker-on-alpine-linux/</a><li><a href=http://dahl-jacobsen.dk/tips/blog/2018-03-15-alpine-on-raspberry-pi/>http://dahl-jacobsen.dk/tips/blog/2018-03-15-alpine-on-raspberry-pi/</a></ul><div class=footnotes role=doc-endnotes><hr><ol><li id=fn:1><p>Running (almost) fully from RAM.&nbsp;<a href=#fnref:1 class=footnote-backref role=doc-backlink>↩︎</a></p><li id=fn:2><p>CF = Compact disk.&nbsp;<a href=#fnref:2 class=footnote-backref role=doc-backlink>↩︎</a></p><li id=fn:3><p>On Linux Id usually opt for <code>dd</code>, on Windows the Raspberry Pi Imager is a sensible choice.&nbsp;<a href=#fnref:3 class=footnote-backref role=doc-backlink>↩︎</a></p><li id=fn:4><p>100MB is overly conservative, but keep in mind I had a very small SD Card, with only 4GiB storage. 250MB or even 500MB should be a more sensible default if you have a bigger SD Card (e.g. 32GiB).&nbsp;<a href=#fnref:4 class=footnote-backref role=doc-backlink>↩︎</a></p><li id=fn:5><p>An alternative is to select <code>data</code> disk mode, but it didnt work for me.&nbsp;<a href=#fnref:5 class=footnote-backref role=doc-backlink>↩︎</a></p><li id=fn:6><p><em>ovl</em> is short for <em>overlay</em>. Not to be confused with <em>vol</em> for <em>volume</em>.&nbsp;<a href=#fnref:6 class=footnote-backref role=doc-backlink>↩︎</a></p></ol></div></div><footer class=post-footer><ul class=post-tags><li><a href=https://thiagowfx.github.io/tags/linux/>linux</a><li><a href=https://thiagowfx.github.io/tags/selfhosted/>selfhosted</a></ul></footer><div style=text-align:center><a href="mailto:tbperrotta@gmail.com?subject=RE: Not%20Just%20Serendipity comment for '%e2%98%85%20Alpine%20Linux%20on%20Raspberry%20Pi%3a%20Diskless%20Mode%20with%20persistent%20storage'" target=_blank><button>Reply via email</button></a></div></article></main><footer class=footer><span>Copyright © 2021 - 2023 Thiago Perrotta • <a href=https://creativecommons.org/licenses/by-nc-sa/4.0/>CC BY-NC-SA 4.0</a><a href=https://thiagowfx.github.io/index.xml>RSS</a></span>
<span>Powered by
<a href=https://gohugo.io/ rel="noopener noreferrer" target=_blank>Hugo</a> &amp;
<a href=https://github.com/adityatelange/hugo-PaperMod/ rel=noopener target=_blank>PaperMod</a></span></footer><a href=#top aria-label="go to top" title="Go to Top (Alt + G)" class=top-link id=top-link accesskey=g style=visibility:visible;opacity:1><svg xmlns=http://www.w3.org/2000/svg viewBox="0 0 12 6" fill=currentcolor><path d="M12 6H0l6-6z"></path></svg></a><script data-template-shadow-root>(()=>{document.currentScript.remove();processNode(document);function processNode(node){node.querySelectorAll("template[shadowroot]").forEach(element=>{let shadowRoot = element.parentElement.shadowRoot;if (!shadowRoot) {try {shadowRoot=element.parentElement.attachShadow({mode:element.getAttribute("shadowroot")});shadowRoot.innerHTML=element.innerHTML;element.remove()} catch (error) {} if (shadowRoot) {processNode(shadowRoot)}}})}})()</script>