summaryrefslogtreecommitdiffstats
path: root/docs/framebuffer
blob: 453ac8f32e7f437b1f27247682abb30738bca02d (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
#!/bin/sh

PREREQ=""
prereqs()
{
	echo "$PREREQ"
}
case $1 in
# get pre-requisites
prereqs)
	prereqs
	exit 0
	;;
esac


# The options part of the kernel "video=" argument (i.e. everyting
# after "video=<fbdriver>:") has very inconsistent rules.
#
# Generally the following applies:
# 1) options are comma-separated
# 2) options can be in either of these three forms:
#    <arg>=<value>, <arg>:<value>, <boolean-arg>.
# 3) the "mode" or "mode_option" option (name depends on the framebuffer driver)
#    has the form <xres>x<yres>[M][R][-<bpp>][@<refresh>][i][m]
#    and may or may not start with "mode=" or "mode_option="
#    -> please adjust as necessary in the parse_video_opts() function
#
# When the options are used with modules, they need to be space-separated
# and the following conversions are needed:
#	<arg>:<value> -> <arg>=<value>
#	<boolean-arg> -> <boolean-arg>=1
#	<modevalue>   -> mode=<modevalue> or mode_option=<modevalue>
parse_video_opts()
{
	local OPTS="$1"
	local IFS=","

	# Must be a line like video=<fbdriver>:<opt1>,[opt2]...
	if [ "${OPTS}" = "${OPTS%%:*}" ]; then
		return
	fi
	OPTS="${OPTS#*:}"
	for opt in ${OPTS}; do
		# Already in the "<arg>=<value>" form
		if [ "${opt}" != "${opt#*=}" ]; then
			printf "%s" "$opt "
		# In the "<arg>:<value>" form
		elif [ "${opt}" != "${opt#*:}" ]; then
			printf "%s" "${opt%:*}=${opt#*:} "
		# Presumably a modevalue without the "mode=" prefix
		elif [ "${opt}" != "${opt#[0-9]*x[0-9]}" ]; then
		  	# Adjust: mode= option?
			printf "%s" "mode=$opt "
			# ... or mode_option= option?
			# printf "%s" "mode_option=$opt "
		# Presumably a boolean
		else
			printf "%s" "${opt}=1 "
		fi
	done
}

FB=""
OPTS=""

# shellcheck disable=SC2013
for x in $(cat /proc/cmdline); do
	case ${x} in
	vga=*)
		FB="vesafb";
		OPTS="";
		;;
	video=*)
		FB=${x#*=}
		FB="${FB%%:*}"
		OPTS="$(parse_video_opts "${x}")"
	esac
done

# Module-specific workarounds
case ${FB} in
matroxfb)
	# Map command line name to module name
	FB=matroxfb_base
	;;
intelfb|i810fb|i915)
	# Needs AGP driver loaded
	/sbin/modprobe intel-agp
	;;
uvesafb)
	# v86d requires /dev/zero and dev/mem, but udev haven't been started yet
	[ -e /dev/zero ] || mknod -m 0666 /dev/zero c 1 5
	[ -e /dev/mem ] || mknod -m 0640 /dev/mem c 1 1
	;;
*)
	;;
esac

if [ -n "${FB}" ]; then
	unset MODPROBE_OPTIONS
	/sbin/modprobe -q fbcon
	# shellcheck disable=SC2086
	/sbin/modprobe -q ${FB} ${OPTS}
fi

if [ -e /proc/fb ]; then
	# shellcheck disable=SC2034
	while read -r fbno desc; do
		if [ $((fbno < 32)) ]; then
			mknod -m 0640 "/dev/fb${fbno}" c 29 "${fbno}"
		fi
	done < /proc/fb
else
	mknod -m 0640 /dev/fb0 c 29 0
fi