Hola. I struggled through this one recently and figured I'd share.
The Overview:
We're looking for an unprivileged LXC to serve as a Jellyfin server. We want to use an Nvidia GPU (in this case a GTX 1070Ti) that is also used by other LXCs for other services. The sharing of the GPU is why we are not considering using VMs. Additionally, this setup uses an NFS share for the video library, which has a couple other requirements.
Phase one: Installing Jellyfin
Phase Two: Adding the GPU.
This part is a pain. The problem is that Jellyfin requires two libraries, libnvcuvid1 and libnvidia-encode1, to work. You can read the documentation here: https://jellyfin.org/docs/general/administration/hardware-acceleration/nvidia#linux-setups
But installing these libraries forces whatever Nvidia drivers your current repositories consider current. This is a problem because the drivers you use on your host must match the drivers you use in the LXC. You have to install the GPU on the host first to be able to install it in the LXC, but you might not end up with the same drivers, so you end up having to go through this process. Do not start doing this yet, just read the bullets below to get an idea of the process:
Here we go:
At this point you should be able to transcode something using the GPU. Set up Nvidia transcoding in the Jellyfin settings, start a movie, and change the resolution to something in the 480 range just force a transcode. Now, on the host, run
The final step: Unlimited encoding streams
Nvidia has a cap on their consumer cards. This can be easily removed using this repo: https://github.com/keylase/nvidia-patch
This needs to be done on both the host and the LXC
Good luck!
The Overview:
We're looking for an unprivileged LXC to serve as a Jellyfin server. We want to use an Nvidia GPU (in this case a GTX 1070Ti) that is also used by other LXCs for other services. The sharing of the GPU is why we are not considering using VMs. Additionally, this setup uses an NFS share for the video library, which has a couple other requirements.
Phase one: Installing Jellyfin
- We'll start by creating a Jellyfin LXC using one of the TTECK scripts available here: https://tteck.github.io/Proxmox/
- Follow the instructions using the host shell in the Proxmox GUI
- I recommend using the Advanced options, so you know what you're getting. I use:
- 2 or 4 cores
- 4096M RAM (two gb should suffice)
- 16G Disk (this will need to be much larger if you plan on many simultaneous transcodes until Jellyfin 10.9 comes out with live transcode clearing)
- Debian 12 (Ubuntu is fine if that's your preference)
- Static IP address
- IP6 Disabled
- Root SSH enabled - You'll need this if you are rebuilding Jellyfin and want to save all your previous data and settings.
- Once created, open the console and stop the jellyfin service
systemctl stop jellyfin
- (Optional - Transfer old settings) If you are transferring from a previous install, you need to transfer the
/var/lib/jellyfin
directory (ignoring the transcodes folder) from the old server to the new one.- WinSCP works well for this, if you remembered to enable root SSH access. If you forgot, open
/etc/ssh/sshd_config
and change the PermitRootLogin option to 'yes' and uncomment it if applicable.systemctl restart sshd
to make it active. - Transfer the directories
- In the new Jellyfin LXC, navigate to /var/lib/jellyfin and fix the ownership with
chown -R jellyfin:jellyfin *
- WinSCP works well for this, if you remembered to enable root SSH access. If you forgot, open
- (Optional - Shared Media folder) Create the mount folder where your shared media library is accessed on the Jellyfin LXC. In this example:
mkdir /mnt/theater
- Shutdown the LXC
- (Optional - Shared Media folder) We need to give the LXC access to your media library. I'm using NFS to do this, so you'll need to make sure whatever server is hosting your media is set up to host NFS shares. We are going to set up the NFS client access on the proxmox host, not the LXC. This will make migrating and snapshotting a bit easier. You can use CIFS, but you'll have to make sure the permissions and user/group settings allow for read and write access.
- On all of your proxmox nodes, create a folder to mount to. In this example we are going to use
/mnt/lxc-share
. It must be the same on all nodes. - Mount the folder to your shared media using the method of your choosing. Test to make sure your files are visible, and that you can add and edit files from the proxmox host. In my case NFS via autofs creates
/mnt/lxc-share/theater
- In the proxmox host, open the LXC configuration file (using LXC 140 in this example)
nano /etc/pve/lxc/140.conf
- Mount the media folder into the LXC using this line at the end of your configuration:
lxc.mount.entry: /mnt/lxc-share/theater mnt/theater none bind 0 0
- We are using this mounting method because it allows for LXC snapshots.
- By mounting the NFS share on the Proxmox host instead of the actual LXC, you have fewer share clients to create and setup. One connection per node, then as many LXCs as you want can access the share using the configuration entry.
- On all of your proxmox nodes, create a folder to mount to. In this example we are going to use
- Start the LXC and confirm.
- In the LXC console, navigate to your mounted media drive (Example:
/mnt/theater
) and confirm that you can see your media and add/edit files. systemctl start jellyfin
Then check to see you are able to navigate to and access jellyfin. If this is a fresh install, you can start setting everything up through Jellyfin, or wait till after we've added the GPU. If this is a transfer, you should be able to log in using your old credentials, but it will ask you to set the admin password. Note, LDAP settings will transfer and work successfully.
- In the LXC console, navigate to your mounted media drive (Example:
- Take a snapshot of the LXC. In this example we are calling the snapshot "post-install."
- Shutdown the LXC
- Did you take a snapshot?
- Seriously, make sure you took a snapshot.
Phase Two: Adding the GPU.
This part is a pain. The problem is that Jellyfin requires two libraries, libnvcuvid1 and libnvidia-encode1, to work. You can read the documentation here: https://jellyfin.org/docs/general/administration/hardware-acceleration/nvidia#linux-setups
But installing these libraries forces whatever Nvidia drivers your current repositories consider current. This is a problem because the drivers you use on your host must match the drivers you use in the LXC. You have to install the GPU on the host first to be able to install it in the LXC, but you might not end up with the same drivers, so you end up having to go through this process. Do not start doing this yet, just read the bullets below to get an idea of the process:
- Install drivers on the host. There is a whole world of how-tos explaining how to install an Nvidia GPU on Proxmox. Get to the point where you can run
nvidia-smi
and see your card listed. I strongly recommend using this process for selecting the drivers: https://github.com/keylase/nvidia-patch#step-by-step-guide. Just the drivers for now, we will address the patch later. Pick some relatively newer drivers. - Make sure you snapshotted your Jellyfin LXC! Install the two libraries on the LXC following the Jellyfin Instructions: https://jellyfin.org/docs/general/administration/hardware-acceleration/nvidia#linux-setups This will download the drivers identified as required packages, and you will be able to see what drivers you need to install on the host. Make note of the driver version.
- Rollback the LXC to the snapshot, removing the drivers and packages.
- Back on the host, uninstall the previous drivers
./NVIDIA-Linux-x86_64-430.50.run --uninstall
(edit for the version you have) and go through the directions for downloading and installing the needed version. - Repeat the steps from the jellyfin instructions to install the two libraries.
- Test to make sure the card is transcoding.
Here we go:
- Install the GPU on your Proxmox host(s). I won't cover this process, you can find the steps online. Use the driver download, not the repository, found here: https://github.com/keylase/nvidia-patch#step-by-step-guide . You will need to disable nuveau, and all sorts of other obnoxious steps. Google "proxmox lxc plex transcode" and you should find guides, just do the host steps, not the LXC steps.
- Shut down the LXC.
- On the host,
ls -l /dev/nvidia*
. We have to pass these to the LXC. Take note of the two numbers separated by a comma, after the group and before the date. You can use both numbers, or just the first number with an asterisk. You should see the following lines, and we are going to pass them all:- /dev/nvidia0
- /dev/nvidiactl
- /dev/nvidia-modset
- /dev/nvidia-uvm
- /dev/nvidia-uvm-tools
- /dev/nvidia-caps/nvidia-cap1
- /dev/nvidia-caps/nvidia-cap2
- To pass these to the LXC you need the following lines in the configuration file
/etc/pve/lxc/140.conf
. Note that you need to replace the bold/underlined numbers with the ones from your system:lxc.cgroup2.devices.allow: c [U][B]195[/B][/U]:* rwm
lxc.cgroup2.devices.allow: c [U][B]234[/B][/U]:* rwm
lxc.cgroup2.devices.allow: c [U][B]235[/B][/U]:* rwm
lxc.cgroup2.devices.allow: c [U][B]238[/B][/U]:* rwm
lxc.mount.entry: /dev/nvidia0 dev/nvidia0 none bind,optional,create=file
lxc.mount.entry: /dev/nvidiactl dev/nvidiactl none bind,optional,create=file
lxc.mount.entry: /dev/nvidia-uvm dev/nvidia-uvm none bind,optional,create=file
lxc.mount.entry: /dev/nvidia-modeset dev/nvidia-modeset none bind,optional,create=file
lxc.mount.entry: /dev/nvidia-uvm-tools dev/nvidia-uvm-tools none bind,optional,create=file
lxc.mount.entry: /dev/nvidia-caps/nvidia-cap1 dev/nvidia-caps/nvidia-cap1 none bind,optional,create=file
lxc.mount.entry: /dev/nvidia-caps/nvidia-cap2 dev/nvidia-caps/nvidia-cap2 none bind,optional,create=file
- Take a snapshot!! "gpu-mounted" or something like that.
- Start the LXC. Install the proprietary drivers as described in the Jellyfin instructions here: https://jellyfin.org/docs/general/administration/hardware-acceleration/nvidia#linux-setups
- https://wiki.debian.org/NvidiaGraphicsDrivers and go to the LXC distribution. I'm using debian 12
- Add "contrib", "non-free" and "non-free-firmware" components to /etc/apt/sources.list
apt update
apt install firmware-misc-nonfree
NOTE: I removed nvidia-driver from the instructions. We will let the next step take care of this.- Back to the jellyfin instructions https://jellyfin.org/docs/general/administration/hardware-acceleration/nvidia#linux-setups
apt install -y jellyfin-ffmpeg5
(Should already be installed)apt install -y libnvcuvid1 libnvidia-encode1
This is where you'll see drivers installed, take note of the version. Mine was 525.147.05-4.
- Now roll back to the "gpu-mounted" snapshot once you noted the correct driver version.
- On the host, uninstall the drivers from step one using the same command you used to install them, but add the
--uninstall
flag. - Download the correct drivers from the site in step one, and install them. Make sure the GPU registers when you run
nvidia-smi
- On the LXC, repeat step 6 again.
- Reboot the LXC
- You'll be able to run nvidia-smi from the LXC, though you won't see any processes (these only show up on the host).
At this point you should be able to transcode something using the GPU. Set up Nvidia transcoding in the Jellyfin settings, start a movie, and change the resolution to something in the 480 range just force a transcode. Now, on the host, run
nvidia-smi
and you should see the transcoding process listed. It'll have "jellyfin-ffmpeg" in the process name. Open up a second window and start another movie simultaneously, drop the resolution, and run nvidia-smi
on the host. You should now see two processes indicating the second transcode is working.The final step: Unlimited encoding streams
Nvidia has a cap on their consumer cards. This can be easily removed using this repo: https://github.com/keylase/nvidia-patch
This needs to be done on both the host and the LXC
- Go to
/opt/nvidia
which might already exist wget https://raw.githubusercontent.com/keylase/nvidia-patch/master/patch.sh
bash ./patch.sh
Good luck!
Last edited: