Commit | Line | Data |
---|---|---|
f617db13 | 1 | #! /bin/bash |
8e08f814 MW |
2 | ### |
3 | ### X startup script | |
f617db13 | 4 | |
8e08f814 MW |
5 | ###-------------------------------------------------------------------------- |
6 | ### Utility functions. | |
f617db13 | 7 | |
8e08f814 MW |
8 | ## Progress indicators. |
9 | info=yes | |
10 | info () { | |
11 | case $info in yes) echo "- $*" >&2 ;; esac | |
12 | } | |
f617db13 | 13 | |
8e08f814 MW |
14 | run=yes |
15 | run () { | |
16 | local what=$1; shift | |
17 | local bg=no | |
f617db13 | 18 | |
8e08f814 MW |
19 | case $what in bg*) bg=yes what=${what#bg} ;; esac |
20 | info "run $what: $*" | |
f617db13 | 21 | |
8e08f814 MW |
22 | case "$run,$bg" in |
23 | yes,no) "$@" ;; | |
24 | yes,yes) "$@" & ;; | |
25 | esac | |
26 | } | |
eebca092 | 27 | |
74bd743a MW |
28 | manage () { |
29 | local when=$(date +%s) now | |
30 | local fail=0 rc report | |
31 | ||
32 | while :; do | |
33 | "$@"; rc=$? | |
34 | case $rc in | |
35 | 0) info "manage $1: successful exit"; break ;; | |
36 | 143) info "manage $1: terminated"; break ;; | |
37 | esac | |
38 | now=$(date +%s) | |
39 | report="rc = $rc" | |
40 | if (( $now - $when > 5 )); then | |
41 | fail=0 | |
42 | else | |
43 | report="$report, early failure" | |
44 | fail=$(( $fail + 1 )) | |
45 | if (( $fail >= 5 )); then | |
46 | info "manage $1: exit ($report), giving up after $fail failures" | |
47 | break | |
48 | fi | |
49 | fi | |
50 | info "manage $1: exit ($report), restarting" | |
51 | when=$now | |
52 | done | |
53 | } | |
54 | ||
8e08f814 | 55 | ## Program choice |
eebca092 | 56 | pick_program () { |
8e08f814 MW |
57 | local what=$1; shift |
58 | local choice=false | |
eebca092 | 59 | for i in "$@"; do |
8e08f814 | 60 | if type -t >/dev/null "$i"; then choice=$i; break; fi |
eebca092 | 61 | done |
8e08f814 MW |
62 | info "pick $what = $choice" |
63 | echo "$choice" | |
eebca092 MW |
64 | } |
65 | ||
8e08f814 MW |
66 | ###-------------------------------------------------------------------------- |
67 | ### Parse arguments. | |
f617db13 | 68 | |
8e08f814 MW |
69 | vnc=no |
70 | atomtag= | |
71 | start=yes | |
72 | wait=yes | |
f617db13 | 73 | |
8e08f814 MW |
74 | for opt; do |
75 | case "$opt" in | |
76 | help) | |
77 | cat <<EOF | |
78 | Options: | |
79 | tag=TAG | |
80 | [no]trace | |
81 | [no]info | |
82 | [no]run | |
83 | [no]start | |
84 | [no]wait | |
85 | [no]vnc | |
86 | EOF | |
87 | exit | |
88 | ;; | |
f617db13 | 89 | |
8e08f814 MW |
90 | tag=*) atomtag=/${opt#tag=} ;; |
91 | trace) set -x ;; | |
92 | notrace) set +x ;; | |
93 | info | run | start | wait | vnc) eval "$opt=yes" ;; | |
94 | noinfo | norun | nostart | nowait | novnc) eval "${opt#no}=no" ;; | |
95 | ||
96 | *) echo "unknown option $opt" >&2; exit 1 ;; | |
97 | esac | |
98 | done | |
99 | ||
100 | ###-------------------------------------------------------------------------- | |
27b6787d MW |
101 | ### Preliminary hook. |
102 | ||
103 | if [ -r $HOME/.xinitrc-prehook ]; then | |
104 | . $HOME/.xinitrc-prehook | |
105 | fi | |
106 | ||
107 | ###-------------------------------------------------------------------------- | |
8e08f814 MW |
108 | ### Iniitial settings. |
109 | ||
110 | ## Assume X sessions are secure. | |
111 | export __mdw_sechost="`hostname`" | |
112 | ||
113 | ## Obtain the screen dimensions. | |
114 | case ",$XWIDTH,$XHEIGHT," in | |
b8d36c3c | 115 | *,,*) eval $(xscsize -bx; xscsize -bmx) ;; |
8e08f814 | 116 | esac |
b8d36c3c MW |
117 | case ",$XNSCR," in |
118 | ,,) | |
119 | XNSCR=1 XSCR0_X=0 XSCR0_Y=0 XSCR0_WIDTH=$XWIDTH XSCR0_HEIGHT=XHEIGHT | |
120 | ;; | |
121 | esac | |
122 | i=0; while (( i < XNSCR )); do | |
123 | eval "x=\$XSCR${i}_X y=\$XSCR${i}_Y | |
124 | wd=\$XSCR${i}_WIDTH ht=\$XSCR${i}_HEIGHT" | |
125 | info "screen #$i = ${wd}x${ht}+${x}+${y}" | |
126 | i=$(( i + 1 )) | |
127 | done | |
8e08f814 MW |
128 | |
129 | initialize () { | |
130 | ## Load the X resource database. | |
131 | run init xrdb -override $HOME/.Xdefaults | |
132 | ||
133 | ## Random xsettery. | |
8ab9747e | 134 | run init xset b 5 2000 50 |
8e08f814 MW |
135 | run init xset r rate 500 50 |
136 | run init xset m 2 1 | |
137 | ||
f4450ac9 | 138 | ## Key mappings. |
cd38b61d MW |
139 | xmodmap $HOME/.xmodmap |
140 | if [ -r $HOME/.xmodmap-local ]; then | |
141 | xmodmap $HOME/.xmodmap-local | |
142 | fi | |
8e08f814 MW |
143 | } |
144 | ||
145 | ###-------------------------------------------------------------------------- | |
146 | ### Start a window manager. | |
147 | ||
6f6bd6da | 148 | wm=$(pick_program window-manager e16 compiz enlightenment e17 twm) |
8e08f814 MW |
149 | wmopts="" |
150 | case "$wm,$vnc" in | |
151 | enlightenment,yes | e16,yes) | |
152 | wmopts="$eopts -econfdir $HOME/.enlightenment-vnc" | |
153 | ;; | |
154 | esac | |
155 | ||
16ee7a8c | 156 | start-e16 () { |
74bd743a | 157 | run bginit manage $wm $wmopts |
16ee7a8c MW |
158 | win=nil |
159 | for i in $(seq 10); do | |
b35a9c36 | 160 | sleep 1 |
16ee7a8c MW |
161 | if eesh version >/dev/null 2>&1; then |
162 | win=t | |
163 | break | |
164 | fi | |
16ee7a8c MW |
165 | done |
166 | case $win in | |
167 | t) | |
168 | info "$wm started ok" | |
169 | run init xsetroot -cursor_name left_ptr | |
170 | ;; | |
171 | nil) | |
172 | info "$wm failed to start!" | |
173 | ;; | |
174 | esac | |
175 | } | |
176 | ||
177 | start-window-manager () { | |
178 | case $(type -t start-$wm || echo "not-found") in | |
179 | function) | |
180 | start-$wm $wmopts | |
181 | ;; | |
182 | *) | |
74bd743a | 183 | run bginit manage $wm $wmopts |
16ee7a8c MW |
184 | ;; |
185 | esac | |
8e08f814 MW |
186 | } |
187 | ||
188 | ###-------------------------------------------------------------------------- | |
189 | ### Random useful clients. | |
190 | ||
54951353 MW |
191 | start-clients-local () { :; } |
192 | ||
8e08f814 | 193 | start-clients () { |
2e60c7f3 | 194 | |
5587cf4f | 195 | ## Gnome session. |
d83c96c9 MW |
196 | case "$vnc,$(xfce4-session --version 2>&1),$(gnome-session --version 2>&1)" |
197 | in | |
198 | no,xfce4-session*) | |
199 | run bginit xfce4-session | |
200 | ;; | |
201 | no,*,gnome-session\ 2.3[2-9].* | \ | |
202 | no,*,gnome-session\ 2.4[0-9].* | \ | |
203 | no,*,gnome-session\ 2.[1-9][0-9][0-9]*) | |
2e60c7f3 MW |
204 | run bginit gnome-session --session mdw |
205 | ;; | |
d83c96c9 | 206 | no,*,gnome-session*) |
2e60c7f3 MW |
207 | run bginit gnome-session |
208 | ;; | |
209 | esac | |
54951353 MW |
210 | |
211 | ## Local clients. | |
212 | start-clients-local | |
8e08f814 MW |
213 | } |
214 | ||
215 | ###-------------------------------------------------------------------------- | |
216 | ### Main screen layout. | |
f617db13 | 217 | |
8e08f814 | 218 | ## Choose appropriate clients. |
384f691a | 219 | emacs=$(pick_program emacs emacs23 emacs24 emacs22 emacs21 emacs) |
8e08f814 | 220 | term=$(pick_program terminal pterm Eterm xterm) |
f617db13 | 221 | |
8e08f814 | 222 | ## Emacs window measurements. |
35f728b2 MW |
223 | ## |
224 | ## e_colwd = basic width of a column in pixels | |
225 | ## e_hextra = extra horizontal width in pixels | |
226 | ## Width of an N-column Emacs frame in pixels will be | |
227 | ## N*e_colwd + e_hextra | |
228 | ## e_colchars = width of a column in Emacs `-geometry' units | |
229 | ## e_cextra = extra horizontal width in Emacs `-geometry' units | |
230 | ## So an N-column frame should be reported to Emacs as being | |
231 | ## N*e_colchars + e_cextra geometry units wide | |
232 | ## e_lineht = height of a character line in pixels | |
233 | ## e_vextra = number of additional vertical cruft pixels | |
234 | ## So an N-line Emacs frame takes N*e_lineht + e_vextra pixels | |
85bbf8d3 | 235 | case "$emacs" in |
8e08f814 | 236 | emacs21 | emacs) |
85bbf8d3 MW |
237 | e_colwd=492 e_hextra=34 |
238 | e_colchars=82 e_cextra=-2 | |
239 | e_lineht=13 e_vextra=52 | |
240 | ;; | |
0c47c911 | 241 | emacs22 | emacs23) |
85bbf8d3 MW |
242 | e_colwd=492 e_hextra=8 |
243 | e_colchars=82 e_cextra=-6 | |
244 | e_lineht=13 e_vextra=46 | |
245 | ;; | |
384f691a MW |
246 | emacs24) |
247 | e_colwd=492 e_hextra=5 | |
248 | e_colchars=82 e_cextra=-6 | |
249 | e_lineht=13 e_vextra=42 | |
250 | ;; | |
85bbf8d3 | 251 | esac |
f617db13 | 252 | |
8e08f814 | 253 | ## Terminal window measurements. |
35f728b2 MW |
254 | ## |
255 | ## t_wd = the window width, in pixels | |
256 | ## t_lineht, t_vextra = height parameters: if the window is N lines high, | |
257 | ## then it will be N*t_lineht + t_vextra pixels high | |
85bbf8d3 | 258 | case "$term" in |
e2b44bd9 | 259 | pterm) t_wd=504 t_lineht=13 t_vextra=23 geom=-geometry;; |
85bbf8d3 MW |
260 | Eterm) t_wd=504 t_lineht=13 t_vextra=23 geom=-g;; |
261 | xterm) t_wd=507 t_lineht=13 t_vextra=27 geom=-geometry;; | |
262 | esac | |
f617db13 | 263 | |
8e08f814 | 264 | ## GNOME stuff measurements. |
6c3d0b23 | 265 | declare -i panelwd=64 xbound=$(( XWIDTH - panelwd )) |
f617db13 | 266 | |
8e08f814 MW |
267 | ## Choose a width for Emacs. |
268 | ## | |
269 | ## We'd like it to be as wide as possible, allowing for a column of xterms | |
270 | ## down the right hand side. However, I'd prefer a double-width Emacs to a | |
271 | ## single-width Emacs and xterms. If it's not going to work at all, a single | |
272 | ## Emacs column will have to do. Also, there's a strange thing with Emacs21 | |
273 | ## and the toolbar, so we add on some rows which are later mysteriously | |
274 | ## subtracted. | |
f617db13 | 275 | |
b8d36c3c MW |
276 | declare -i lim=XSCR0_WIDTH |
277 | if (( lim > xbound )); then lim=xbound; fi | |
7f903e21 | 278 | declare -i ecols=$(( (lim - t_wd - e_hextra)/e_colwd )) |
b8d36c3c MW |
279 | if (( ecols < 2 && lim > 2*e_colwd + e_hextra )); then ecols=2 |
280 | elif (( ecols < 1 )); then ecols=1 | |
f617db13 MW |
281 | fi |
282 | ||
8e08f814 | 283 | declare -i \ |
7f903e21 MW |
284 | emacsx=$(( ecols*e_colchars + e_cextra )) \ |
285 | emacsy=$(( (XHEIGHT - e_vextra)/e_lineht )) | |
8e08f814 MW |
286 | |
287 | start-emacs () { | |
bf057abe | 288 | GDK_NATIVE_WINDOWS=1 run bgclients noip \ |
e10a95ab | 289 | $emacs -bg black -geometry ${emacsx}x${emacsy}+${XSCR0_X}+${XSCR0_Y} |
8e08f814 MW |
290 | } |
291 | ||
292 | ## Now place some xterms. | |
293 | ## | |
294 | ## A few smaller xterms are in general better than one great big one. 35 | |
295 | ## lines is a good height for most terminals. 25 lines is a minimum. The | |
296 | ## strategy for doling out xterms into a column is to make as many 35-liners | |
297 | ## as we can, until the remaining space would be too small for a 25-liner. | |
298 | ## If we can get two 25s out of that then we do (largest first); otherwise | |
299 | ## just make one big one. We stop at the end of a page, once we've made | |
300 | ## three xterms. | |
301 | ||
302 | start-xterms () { | |
303 | ||
304 | ## Initialize some parameters. | |
7f903e21 | 305 | declare -i x=$(( ecols*e_colwd + e_hextra + XSCR0_X )) xb=xbound |
b8d36c3c | 306 | declare -i n=0 pgx=0 l h y ht scr=0 ll=lim |
7f903e21 MW |
307 | declare -i hstd=$(( 35*t_lineht + t_vextra )) |
308 | declare -i hmin=$(( 25*t_lineht + t_vextra )) | |
b8d36c3c MW |
309 | declare -i scrx scry scrwd scrht |
310 | ||
311 | eval "scrx=\$XSCR${scr}_X scry=\$XSCR${scr}_Y | |
312 | scrwd=\$XSCR${scr}_WIDTH scrht=\$XSCR${scr}_HEIGHT" | |
8e08f814 MW |
313 | |
314 | ## Do the placement. | |
315 | while :; do | |
316 | ||
317 | ## Start a new iteration. | |
7f903e21 | 318 | if (( x + t_wd > ll )); then |
b8d36c3c MW |
319 | scr=$(( scr + 1 )) |
320 | if (( scr >= XNSCR )); then | |
321 | if (( n >= 3 )); then break; fi | |
322 | pgx=$(( pgx + XWIDTH )) xb=$(( xb + XWIDTH )) | |
323 | scr=0 | |
324 | fi | |
325 | eval "scrx=\$XSCR${scr}_X scry=\$XSCR${scr}_Y | |
326 | scrwd=\$XSCR${scr}_WIDTH scrht=\$XSCR${scr}_HEIGHT" | |
327 | x=$(( pgx + scrx )) | |
328 | ll=$(( x + scrwd )) | |
329 | if (( ll > xb )); then ll=xb; fi | |
8e08f814 MW |
330 | fi |
331 | ||
332 | ## Make large xterms. | |
b8d36c3c | 333 | y=scry ht=scrht |
7f903e21 | 334 | while (( ht - hstd >= hmin )); do |
8e08f814 | 335 | run bgclients $term $geom 80x35+$x+$y |
7f903e21 | 336 | y=$(( y + hstd )) ht=$(( ht - hstd )) n=$(( n + 1 )) |
8e08f814 MW |
337 | done |
338 | ||
339 | ## Fill the remaining space. | |
7f903e21 MW |
340 | if (( ht >= 2*hmin )); then h=$(( ht - hmin )); else h=ht; fi |
341 | l=$(( (h - t_vextra)/t_lineht )) h=$(( l*t_lineht + t_vextra )) | |
8e08f814 | 342 | run bgclients $term $geom 80x$l+$x+$y |
7f903e21 | 343 | y=$(( y + h )) ht=$(( ht - h )) n=$(( n + 1 )) |
8e08f814 MW |
344 | if ((ht >= hmin)); then |
345 | run bgclients $term $geom 80x25+$x+$y | |
7f903e21 | 346 | n=$(( n + 1 )) |
8e08f814 | 347 | fi |
7f903e21 | 348 | x=$(( x + t_wd )) |
f617db13 | 349 | done |
8e08f814 | 350 | } |
f617db13 | 351 | |
8e08f814 | 352 | ###-------------------------------------------------------------------------- |
3bdada49 MW |
353 | ### Requesters. |
354 | ||
355 | req () { | |
356 | declare title=$1 hist=$2; shift 2 | |
3f88bd1f | 357 | cmd=$(xgetline -t "$title" -p "_Command:" -Hl "$HOME/$hist") && |
3bdada49 MW |
358 | exec "$@" "$cmd" |
359 | } | |
360 | ||
361 | ###-------------------------------------------------------------------------- | |
8e08f814 MW |
362 | ### Final waiting. |
363 | ||
364 | atom=XINIT_COMMAND$atomtag | |
365 | ||
366 | xwait () { | |
367 | while :; do | |
368 | xatom delete $atom | |
369 | info "waiting on $atom" | |
370 | line=$(xatom wait $atom) | |
371 | info "xatom: $line" | |
372 | ||
373 | case "$line" in | |
374 | :help) | |
375 | xmsg -I -t "xinitrc help" -d "xinitrc commands" - <<EOF & | |
376 | :help | |
377 | :emacs :xterms :window-manager :clients | |
3bdada49 | 378 | :ask-run :ask-command |
8e08f814 MW |
379 | :init |
380 | :terminal | |
381 | ! SHELL-COMMAND | |
382 | CLIENT | |
383 | EOF | |
384 | ;; | |
385 | :emacs | :xterms | :window-manager | :clients) | |
386 | start-${line#:} | |
387 | ;; | |
388 | :terminal) | |
389 | run bgclients $term | |
390 | ;; | |
391 | :init) | |
392 | initialize | |
393 | ;; | |
394 | :exec) | |
395 | info "restarting xinitrc" | |
396 | exec "$0" wait nostart | |
397 | ;; | |
3bdada49 | 398 | :ask-run) |
a6780078 | 399 | req "Shell command" .cmd.hist xcatch -F"Fixed 13" -- sh -c& |
3bdada49 MW |
400 | ;; |
401 | :ask-command) | |
402 | req "xinit command" .xinit.hist xatom set XINIT_COMMAND$atomtag& | |
403 | ;; | |
8e08f814 MW |
404 | :*) |
405 | xmsg -E -t "xinitrc error" "Unknown command \`$line'" & | |
406 | ;; | |
407 | !*) | |
408 | eval "${line#!}" | |
409 | ;; | |
410 | *) | |
2f616761 | 411 | set -- $line |
8e08f814 MW |
412 | run bgclients "$@" |
413 | ;; | |
414 | esac | |
415 | done | |
416 | } | |
417 | ||
418 | ###-------------------------------------------------------------------------- | |
8ca91005 MW |
419 | ### Gnome session care and feeding. |
420 | ||
421 | session-running-p () { | |
422 | dbus-send --session --print-reply --dest=org.freedesktop.DBus / \ | |
423 | org.freedesktop.DBus.GetNameOwner string:org.gnome.SessionManager \ | |
424 | >/dev/null 2>&1 | |
425 | } | |
426 | ||
d83c96c9 MW |
427 | dbus-service-running-p () { |
428 | dbus-send >/dev/null 2>&1 --session --print-reply \ | |
429 | --dest=org.freedesktop.DBus / \ | |
430 | org.freedesktop.DBus.GetNameOwner string:$1 | |
431 | } | |
432 | ||
8ca91005 | 433 | kill-gnome-session () { |
d83c96c9 MW |
434 | win=nil |
435 | while read service object logout; do | |
436 | if dbus-service-running-p $service; then win=t; break; fi | |
437 | done <<EOF | |
438 | org.xfce.SessionManager /org/xfce/SessionManager org.xfce.Session.Manager.Shutdown uint32:1 boolean:false | |
439 | org.gnome.SessionManager /org/gnome/SessionManager org.gnome.SessionManager.Logout uint32:2 | |
440 | EOF | |
441 | case $win in nil) return ;; esac | |
442 | info "killing session manager" | |
443 | dbus-send --session --dest=$service $object $logout | |
444 | for i in 1 2 3 4 5; do | |
445 | sleep 1 | |
446 | if ! dbus-service-running-p $service; then break; fi | |
447 | done | |
8ca91005 MW |
448 | } |
449 | ||
450 | ###-------------------------------------------------------------------------- | |
8e08f814 MW |
451 | ### Actually start things up. |
452 | ||
54951353 MW |
453 | if [ -f $HOME/.xinitrc-local ]; then |
454 | . $HOME/.xinitrc-local | |
455 | fi | |
456 | ||
8e08f814 MW |
457 | case "$start" in |
458 | yes) | |
459 | info "starting standard clients" | |
460 | initialize | |
461 | start-window-manager | |
462 | start-clients | |
463 | start-emacs | |
464 | start-xterms | |
465 | ;; | |
466 | no) | |
467 | info "not starting standard clients" | |
468 | ;; | |
469 | esac | |
470 | ||
471 | case "$wait" in | |
472 | yes) | |
473 | xwait | |
8ca91005 | 474 | kill-gnome-session |
8e08f814 MW |
475 | ;; |
476 | no) | |
477 | info "not waiting before exit" | |
478 | ;; | |
479 | esac | |
f617db13 | 480 | |
8e08f814 | 481 | ###----- That's all, folks -------------------------------------------------- |