initial commit

master
Dessa Simpson 2023-10-25 18:25:51 -07:00
commit 99a66f4cfd
15 changed files with 287 additions and 0 deletions

7
.gitignore vendored Normal file
View File

@ -0,0 +1,7 @@
*.xz
*.dsk
*.tgz
# generated from template
k8s/deployment.yaml
k8s/metallb.yaml

13
README.md Normal file
View File

@ -0,0 +1,13 @@
# PDP-11 in Docker/Kubernetes
## Docker
To build the container, run `docker/build.sh [tag]` with the tag you want to apply to the image.
To run the container, use the following command adapted to your needs:
docker run -d --rm --privileged --name pdp -v `pwd`/rq0.dsk:/mnt/rq0.dsk <tag>
## Kubernetes
To run in Kubernetes, first you need to build the docker image and push it to a registry your cluster has access to.
Once you've done that, edit k8s/deployment.yaml

40
bootstrap-k8s.sh Executable file
View File

@ -0,0 +1,40 @@
#!/bin/bash
set -e
usage () {
echo "usage: $0 tag disk_filename"
exit 1
}
[[ $# -eq 2 ]] || usage
tag="$1"
disk_filename="$2"
cd "$(dirname $0)"
[[ -f "$disk_filename" && -r "$disk_filename" ]] || { echo "$disk_filename does not exist, is not a regular file, or is not readable"; exit 1; }
[[ "$disk_filename" == *.xz ]] || { echo "$disk_filename does not appear to be an xz-compressed disk image"; exit 1; }
sed "s!<TAG>!${tag}!" ./k8s/deployment.yaml.template > ./k8s/deployment.yaml
echo "Creating PVC..."
kubectl apply -f ./k8s/pvc.yaml
echo "Creating temporary pod..."
kubectl apply -f ./k8s/pod-tmp.yaml
echo "Waiting for temporary pod..."
while ! kubectl get pod pdp-tmp|grep -q Running; do sleep 1; done
echo "Checking for disk image..."
if ! kubectl exec -it pdp-tmp -- test -f /mnt/rq0.dsk &>/dev/null; then
echo "Uploading compressed disk image..."
kubectl exec pdp-tmp -- rm -f /mnt/rq0.dsk.xz
kubectl cp "$disk_filename" pdp-tmp:/mnt/rq0.dsk.xz
echo "Extracting disk image..."
kubectl exec pdp-tmp -- apk add xz
kubectl exec pdp-tmp -- xz -d /mnt/rq0.dsk.xz
fi
echo "Removing temporary pod..."
kubectl delete -f ./k8s/pod-tmp.yaml
echo "Creating PDP deployment..."
kubectl apply -f ./k8s/deployment.yaml
echo "Creating services..."
for i in ./k8s/svc-*.yaml; do kubectl apply -f "$i"; done
echo "Done! Optionally, run ./k8s/configure-metallb.sh to put the services on an external IP."

13
docker/Dockerfile Normal file
View File

@ -0,0 +1,13 @@
FROM debian:11-slim AS builder
RUN apt-get -y update && apt-get -y upgrade && apt-get -y install gcc libpcap-dev libvdeplug-dev libpcre3-dev libedit-dev libsdl2-dev libpng-dev libsdl2-ttf-dev build-essential && apt-get clean
ADD simh-3.9-0.tgz /
WORKDIR /simh-3.9-0
RUN make pdp11
FROM debian:11-slim
ENV DISK_FILENAME=rq0.dsk
RUN apt-get -y update && apt-get -y upgrade && apt-get -y install ed libpcap-dev iproute2 iptables gcc vdeplug libpcre3 net-tools telnet
COPY --from=builder /simh-3.9-0/BIN/pdp11 /usr/local/bin/pdp11
COPY boot.ini.template /
COPY startup.sh /
CMD /startup.sh

42
docker/boot.ini.template Normal file
View File

@ -0,0 +1,42 @@
; model a PDP-11/70 with maximum memory, fpu, no CIS
set cpu 11/70 4096K fpp
; disable all extra devices by default, will enable later
detach all
reset all
; use 7b ascii terminal
set tto 7b
; set the boot disk as an MSCP UDA50
set rq enabled
set rq0 rauser=1000
attach rq0 /mnt/<DISK_FILENAME>
show rq0
; set one DZ11 8 line async mux
; accessible via telnet to port 4000
set dz enabled
set dz lines=8
set dz 7b
attach -am dz 4000
show dz
; set one DELUA/DEUNA unibus enet controller
; ==> must setup ethX per your system config
set xu enabled
set xu type=delua
attach xu tap:pdp
show xu
; show our config
show devices
show cpu iospace
; Configure all csr/vector
set cpu autoconfig
; Change DZ vector
set dz vector=310
; boot it
boot rq0
<EXIT>

17
docker/build.sh Executable file
View File

@ -0,0 +1,17 @@
#!/bin/sh
usage () {
echo "usage: $0 tag"
exit 1
}
[ -n "$1" ] && tag="$1" || usage
file="simh-3.9-0.tgz"
url="https://codeload.github.com/simh/simh/tar.gz/refs/tags/v3.9-0"
cd "$(dirname $0)"
[ -f "$file" ] || wget "$url" -O "$file"
docker build -t "$tag" .

25
docker/startup.sh Executable file
View File

@ -0,0 +1,25 @@
#!/bin/sh
set -ex
[ -n "$DISK_FILENAME" ] || DISK_FILENAME="rq0.dsk"
[ -f "/mnt/$DISK_FILENAME" ] || { echo "Unable to find tape /mnt/$DISK_FILENAME"; exit 1; }
[ -w "/mnt/$DISK_FILENAME" ] || { echo "Tape /mnt/$DISK_FILENAME is not writable"; exit 1; }
# Set exit command only if NOEXIT not set
[ -z "$NOEXIT" ] && EXIT_CMD="exit" || EXIT_CMD=""
sed -e "s/<DISK_FILENAME>/${DISK_FILENAME}/" \
-e "s/<EXIT>/${EXIT_CMD}/" /boot.ini.template > /boot.ini
cidr=28
hostip=1.3.3.1
pdpip=1.3.3.7
ip tuntap add pdp mode tap
ip link set pdp up
ip addr add "${hostip}/${cidr}" dev pdp
iptables -t nat -A PREROUTING ! -i pdp -p tcp --dport 4000 -j ACCEPT
iptables -t nat -A PREROUTING ! -i pdp -j DNAT --to-destination 1.3.3.7
iptables -t nat -A POSTROUTING ! -o pdp -j MASQUERADE
pdp11 /boot.ini

23
k8s/configure-metallb.sh Executable file
View File

@ -0,0 +1,23 @@
#!/bin/bash
set -e
usage () {
echo "usage: $0 ip"
exit 1
}
[[ $# -eq 1 ]] || usage
ip="$1"
cd "$(dirname $0)"
sed "s!<IP>!${ip}!" ./metallb.yaml.template > ./metallb.yaml
echo "Configuring metallb..."
kubectl apply -f ./metallb.yaml
echo "Configuring services..."
for i in ./svc-*.yaml; do
svc="$(grep -Po "name: \Kpdp-.*$" "$i")"
kubectl patch svc "$svc" -p '{"metadata":{"annotations":{"metallb.universe.tf/allow-shared-ip":"'"$ip"'"}},"spec":{"type": "LoadBalancer","loadBalancerIP":"'"$ip"'"}}'
done
echo "Done!"

View File

@ -0,0 +1,29 @@
apiVersion: apps/v1
kind: Deployment
metadata:
name: pdp
labels:
app.kubernetes.io/name: pdp
spec:
replicas: 1
selector:
matchLabels:
app.kubernetes.io/name: pdp
template:
metadata:
labels:
app.kubernetes.io/name: pdp
spec:
volumes:
- name: disk
persistentVolumeClaim:
claimName: pdp-disk
containers:
- name: pdp
image: <TAG>
volumeMounts:
- name: disk
mountPath: /mnt
securityContext:
allowPrivilegeEscalation: true
privileged: true

15
k8s/metallb.yaml.template Normal file
View File

@ -0,0 +1,15 @@
apiVersion: metallb.io/v1beta1
kind: IPAddressPool
metadata:
name: pdp
namespace: metallb-system
spec:
addresses:
- <IP>/32
autoAssign: false
---
apiVersion: metallb.io/v1beta1
kind: L2Advertisement
metadata:
name: default-l2
namespace: metallb-system

17
k8s/pod-tmp.yaml Normal file
View File

@ -0,0 +1,17 @@
apiVersion: v1
kind: Pod
metadata:
name: pdp-tmp
spec:
volumes:
- name: disk
persistentVolumeClaim:
claimName: pdp-disk
containers:
- name: pdp
image: alpine
command: ["sleep", "infinity"]
volumeMounts:
- name: disk
mountPath: /mnt
restartPolicy: Never

10
k8s/pvc.yaml Normal file
View File

@ -0,0 +1,10 @@
apiVersion: v1
kind: PersistentVolumeClaim
metadata:
name: pdp-disk
spec:
accessModes:
- ReadWriteOnce
resources:
requests:
storage: 1Gi

12
k8s/svc-ftp.yaml Normal file
View File

@ -0,0 +1,12 @@
apiVersion: v1
kind: Service
metadata:
name: pdp-ftp
labels:
app.kubernetes.io/name: pdp
spec:
selector:
app.kubernetes.io/name: pdp
ports:
- protocol: TCP
port: 21

12
k8s/svc-http.yaml Normal file
View File

@ -0,0 +1,12 @@
apiVersion: v1
kind: Service
metadata:
name: pdp-http
labels:
app.kubernetes.io/name: pdp
spec:
selector:
app.kubernetes.io/name: pdp
ports:
- protocol: TCP
port: 80

12
k8s/svc-telnet.yaml Normal file
View File

@ -0,0 +1,12 @@
apiVersion: v1
kind: Service
metadata:
name: pdp-telnet
labels:
app.kubernetes.io/name: pdp
spec:
selector:
app.kubernetes.io/name: pdp
ports:
- protocol: TCP
port: 23