VirtualBox

source: vbox/trunk/src/VBox/Installer/linux/VBoxHeadlessXOrg.sh@ 43876

Last change on this file since 43876 was 43832, checked in by vboxsync, 12 years ago

Installer/linux: HeadlessXOrg clean-up.

  • Property svn:eol-style set to LF
  • Property svn:executable set to *
  • Property svn:keywords set to Author Date Id Revision
File size: 10.6 KB
Line 
1#!/bin/sh
2# $Id: VBoxHeadlessXOrg.sh 43832 2012-11-07 14:37:42Z vboxsync $
3#
4# VirtualBox X Server auto-start service.
5#
6# Copyright (C) 2012 Oracle Corporation
7#
8# This file is part of VirtualBox Open Source Edition (OSE), as
9# available from http://www.virtualbox.org. This file is free software;
10# you can redistribute it and/or modify it under the terms of the GNU
11# General Public License (GPL) as published by the Free Software
12# Foundation, in version 2 as it comes in the "COPYING" file of the
13# VirtualBox OSE distribution. VirtualBox OSE is distributed in the
14# hope that it will be useful, but WITHOUT ANY WARRANTY of any kind.
15#
16
17PATH=$PATH:/bin:/sbin:/usr/sbin
18
19## Start one or several X servers in the background for use with headless
20# rendering. For details, options and configuration see the usage() function
21# further down.
22#
23# I have tried to follow the best practices I could find for writing a Linux
24# service (and doing it in shell script) which should work well with
25# traditional and modern service systems using minimal init or service files.
26# In our case this boils down to:
27# * Start with a single command line, stop using one of ${EXIT_SIGNALS} below.
28# * Stopping with a signal can be done safely using the pid stored in the
29# pid-file and our (presumably unique) command name. For this reason we
30# only support running one instance of the service though.
31# * Start in the foreground. Systems without proper service control can take
32# care of the backgrounding in the init script.
33# * Clean up all sub-processes (X servers) ourselves when we are stopped
34# cleanly and don't provide any other way to clean them up automatically (in
35# case we are stopped uncleanly) as we don't know of a generic safe way to
36# do so, though some service management systems (i.e. systemd) can do so.
37# (A more thorough automatic clean-up would be possible if Xorg didn't
38# potentially have to be run as root, so that we could run all processes
39# using a service-specific user account and just terminate all processes
40# run by that user to clean up.)
41
42## Default configuration file name.
43# @note This is not very nice - /etc/default is actually Debian-specific.
44CONFIGURATION_FILE=/etc/default/virtualbox
45## The name of this script.
46SCRIPT_NAME="$0"
47## Command line we were called with.
48SCRIPT_COMMAND_LINE="$0 $@"
49## The service name. Should match the init script name.
50SERVICE_NAME="vboxheadlessxorg"
51## The descriptive service name.
52SERVICE_LONG_NAME="VBoxHeadless X Server service"
53## Signals and conditions which may be used to terminate the service.
54EXIT_SIGNALS="EXIT HUP INT QUIT ABRT TERM"
55## The default X server configuration directory.
56DEFAULT_CONFIGURATION_FOLDER="/etc/vbox/headlessxorg.conf.d"
57
58## Print usage information for the service script.
59## @todo Perhaps we should support some of the configuration file options from
60# the command line. Opinions welcome.
61## @todo Possibly extract this information for the user manual.
62usage() {
63 cat << EOF
64Usage:
65
66 $(basename "${SCRIPT_NAME}") [<options>]
67
68Start one or several X servers in the background for use with headless
69rendering. We only support X.Org Server at the moment. Starting the X servers
70is managed by dropping one or more files xorg.conf.<n> into a configuration
71directory, by default ${DEFAULT_CONFIGURATION_FOLDER}, but this can be overridden
72in the configuration file (see below). We will attempt to start an X server
73process for each configuration file using display number <n>.
74
75Options:
76
77 -c|--conf-file) Specify an alternative locations for the configuration
78 file. The default location is "${CONFIGURATION_FILE}".
79
80 --help|--usage Print this text.
81
82The optional configuration file should contain a series of lines of the form
83"KEY=value". It will be read in as a command shell sub-script. Here is the
84current list of possible key settings with a short explanation. Usually it
85should be sufficient to change the value of \${HEADLESS_X_ORG_USERS} and to
86leave all other settings unchanged.
87
88 HEADLESS_X_ORG_CONFIGURATION_FOLDER
89 The folder where the configuration files for the X servers are to be found.
90
91 HEADLESS_X_ORG_LOG_FOLDER
92 The folder where log files will be created.
93
94 HEADLESS_X_ORG_LOG_FILE
95 The main log file name.
96
97 HEADLESS_X_ORG_RUN_FOLDER
98 The folder to store run-time data in.
99
100 HEADLESS_X_ORG_CHECK_PREREQUISITES
101 Shell command to execute to check whether all dependencies for the X
102 servers are available - usually a test for a device node. This will be
103 repeated at regular intervals until it returns successfully, so a command
104 which can be executed internally be the shell (like "[") is preferable.
105 The default command waits until the udev event queue has settled.
106
107 HEADLESS_X_ORG_USERS
108 List of users who will have access to the X servers started and for whom we
109 will provide the configuration details via VirtualBox extra data. This
110 variable is only used by the commands in the default configuration
111 (\${HEADLESS_X_ORG_SERVER_PRE_COMMAND} and
112 \${HEADLESS_X_ORG_SERVER_POST_COMMAND}), and not by the service itself.
113
114 HEADLESS_X_ORG_SERVER_PRE_COMMAND
115 Command to execute once to perform any set-up needed before starting the
116 X servers, such as setting up the X server authentication. The default
117 command creates an authority file for each of the users in the list
118 \${HEADLESS_X_ORG_USERS}.
119
120 HEADLESS_X_ORG_SERVER_COMMAND
121 The default X server start-up command. It will be passed three parameters
122 - in order, the screen number to use, the path of the X.Org configuration
123 file to use and the path of the X server log file to create.
124
125 HEADLESS_X_ORG_SERVER_POST_COMMAND
126 Command to execute once the X servers have been successfully started. By
127 default this stores the service configuration information to VirtualBox
128 extra data for each of the users in the list \${HEADLESS_X_ORG_USERS}.
129 It will be passed a single parameter which is a space-separated list of the
130 X server screen numbers.
131EOF
132}
133
134# Default configuration.
135HEADLESS_X_ORG_CONFIGURATION_FOLDER="${DEFAULT_CONFIGURATION_FOLDER}"
136HEADLESS_X_ORG_LOG_FOLDER="/var/log/${SERVICE_NAME}"
137HEADLESS_X_ORG_LOG_FILE="${SERVICE_NAME}.log"
138HEADLESS_X_ORG_RUN_FOLDER="/var/run/${SERVICE_NAME}"
139HEADLESS_X_ORG_CHECK_PREREQUISITES="udevadm settle || ! udevadm -V"
140HEADLESS_X_ORG_USERS=""
141
142X_AUTH_FILE="${HEADLESS_X_ORG_RUN_FOLDER}/xauth"
143default_pre_command()
144{
145 echo > "${X_AUTH_FILE}"
146 key="$(dd if=/dev/urandom count=1 bs=16 2>/dev/null | od -An -x)"
147 xauth -f "${X_AUTH_FILE}" add :0 . "${key}"
148 for i in ${HEADLESS_X_ORG_USERS}; do
149 cp "${X_AUTH_FILE}" "${X_AUTH_FILE}.${i}"
150 chown "${i}" "${X_AUTH_FILE}.${i}"
151 done
152}
153HEADLESS_X_ORG_SERVER_PRE_COMMAND="default_pre_command"
154
155default_command()
156{
157 auth="${HEADLESS_X_ORG_RUN_FOLDER}/xauth"
158 # screen=$1
159 # conf_file=$2
160 # log_file=$3
161 Xorg :"${1}" -auth "${auth}" -config "${2}" -logverbose 0 -logfile /dev/null -verbose 7 > "${3}" 2>&1
162}
163HEADLESS_X_ORG_SERVER_COMMAND="default_command"
164
165default_post_command()
166{
167 # screens=$1
168 for i in ${HEADLESS_X_ORG_USERS}; do
169 su ${i} -c "VBoxManage setextradata global HeadlessXServer/Screens \"${1}\""
170 su ${i} -c "VBoxManage setextradata global HeadlessXServer/AuthFile \"${HEADLESS_X_ORG_RUN_FOLDER}/xauth\""
171 done
172}
173HEADLESS_X_ORG_SERVER_POST_COMMAND="default_post_command"
174
175## The function definition at the start of every non-trivial shell script!
176abort() {
177 ## $@, ... Error text to output to standard error in printf format.
178 printf "$@" >&2
179 exit 1
180}
181
182## Milder version of abort, when we can't continue because of a valid condition.
183abandon() {
184 ## $@, ... Text to output to standard error in printf format.
185 printf "$@" >&2
186 exit 0
187}
188
189abort_usage() {
190 usage >&2
191 abort "$@"
192}
193
194# Print a banner message
195banner() {
196 cat << EOF
197${VBOX_PRODUCT} VBoxHeadless X Server start-up service Version ${VBOX_VERSION_STRING}
198(C) 2005-${VBOX_C_YEAR} ${VBOX_VENDOR}
199All rights reserved.
200
201EOF
202}
203
204# Get the directory where the script is located.
205VBOX_FOLDER="$(dirname "${SCRIPT_NAME}")"
206VBOX_FOLDER=$(cd "${VBOX_FOLDER}" && pwd)
207[ -d "${VBOX_FOLDER}" ] ||
208 abort "Failed to change to directory ${VBOX_FOLDER}.\n"
209# And change to the root directory so we don't hold any other open.
210cd /
211
212[ -r "${VBOX_FOLDER}/scripts/generated.sh" ] ||
213 abort "${LOG_FILE}" "Failed to find installation information in ${VBOX_FOLDER}.\n"
214. "${VBOX_FOLDER}/scripts/generated.sh"
215
216# Parse our arguments.
217while [ "$#" -gt 0 ]; do
218 case $1 in
219 -c|--conf-file)
220 [ "$#" -gt 1 ] ||
221 {
222 banner
223 abort "%s requires at least one argument.\n" "$1"
224 }
225 CONFIGURATION_FILE="$2"
226 shift
227 ;;
228 --help|--usage)
229 banner
230 usage
231 exit 0
232 ;;
233 *)
234 banner
235 abort_usage "Unknown argument $1.\n"
236 ;;
237 esac
238 shift
239done
240
241[ -r "${CONFIGURATION_FILE}" ] && . "${CONFIGURATION_FILE}"
242
243# If something fails here we will catch it when we create the directory.
244[ -e "${HEADLESS_X_ORG_LOG_FOLDER}" ] &&
245 [ -d "${HEADLESS_X_ORG_LOG_FOLDER}" ] &&
246 rm -rf "${HEADLESS_X_ORG_LOG_FOLDER}.old" 2> /dev/null &&
247mv "${HEADLESS_X_ORG_LOG_FOLDER}" "${HEADLESS_X_ORG_LOG_FOLDER}.old" 2> /dev/null
248mkdir -p "${HEADLESS_X_ORG_LOG_FOLDER}" 2>/dev/null ||
249{
250 banner
251 abort "Failed to create log folder \"${HEADLESS_X_ORG_LOG_FOLDER}\".\n"
252}
253mkdir -p "${HEADLESS_X_ORG_RUN_FOLDER}" 2>/dev/null ||
254{
255 banner
256 abort "Failed to create run folder \"${HEADLESS_X_ORG_RUN_FOLDER}\".\n"
257}
258exec > "${HEADLESS_X_ORG_LOG_FOLDER}/${HEADLESS_X_ORG_LOG_FILE}" 2>&1
259
260banner
261
262# Wait for our dependencies to become available. The increasing delay is
263# probably not the cleverest way to do this.
264DELAY=1
265while ! eval ${HEADLESS_X_ORG_CHECK_PREREQUISITES}; do
266 sleep $((${DELAY} / 10 + 1))
267 DELAY=$((${DELAY} + 1))
268done
269
270# Do any pre-start setup.
271eval "${HEADLESS_X_ORG_SERVER_PRE_COMMAND}"
272
273X_SERVER_PIDS=""
274X_SERVER_SCREENS=""
275trap "kill \${X_SERVER_PIDS} 2>/dev/null" ${EXIT_SIGNALS}
276space="" # Hack to put spaces between the pids but not before or after.
277for conf_file in "${HEADLESS_X_ORG_CONFIGURATION_FOLDER}"/*; do
278 [ x"${conf_file}" = x"${HEADLESS_X_ORG_CONFIGURATION_FOLDER}/*" ] &&
279 ! [ -e "${conf_file}" ] &&
280 abort "No configuration files found.\n"
281 filename="$(basename "${conf_file}")"
282 screen="$(expr "${filename}" : "xorg\.conf\.\(.*\)")"
283 [ 0 -le "${screen}" ] 2>/dev/null ||
284 abort "Badly formed file name \"${conf_file}\".\n"
285 log_file="${HEADLESS_X_ORG_LOG_FOLDER}/Xorg.${screen}.log"
286 eval "${HEADLESS_X_ORG_SERVER_COMMAND} \"\${screen}\" \"\${conf_file}\" \"\${log_file}\"" "&"
287 X_SERVER_PIDS="${X_SERVER_PIDS}${space}$!"
288 X_SERVER_SCREENS="${X_SERVER_SCREENS}${space}${screen}"
289 space=" "
290done
291
292# Do any post-start work.
293eval "${HEADLESS_X_ORG_SERVER_POST_COMMAND} \"${X_SERVER_SCREENS}\""
294
295wait
Note: See TracBrowser for help on using the repository browser.

© 2024 Oracle Support Privacy / Do Not Sell My Info Terms of Use Trademark Policy Automated Access Etiquette