aboutsummaryrefslogtreecommitdiff
path: root/en_US.ISO8859-1/articles/rc-scripting/article.sgml
blob: d763a65c2a88991e510509fb678bcbc01ece174a (plain) (blame)
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
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
561
562
563
564
565
566
567
568
569
570
571
572
573
574
575
576
577
578
579
580
581
582
583
584
585
586
587
588
589
590
591
592
593
594
595
596
597
598
599
600
601
602
603
604
605
606
607
608
609
610
611
612
613
614
615
616
617
618
619
620
621
622
623
624
625
626
627
628
629
630
631
632
633
634
635
636
637
638
639
640
641
642
643
644
645
646
647
648
649
650
651
652
653
654
655
656
657
658
659
660
661
662
663
664
665
666
667
668
669
670
671
672
673
674
675
676
677
678
679
680
681
682
683
684
685
686
687
688
689
690
691
692
693
694
695
696
697
698
699
700
701
702
703
704
705
706
707
708
709
710
711
712
713
714
715
716
717
718
719
720
721
722
723
724
725
726
727
728
729
730
731
732
733
734
735
736
737
738
739
740
741
742
743
744
745
746
747
748
749
750
751
752
753
754
755
756
757
758
759
760
761
762
763
764
765
766
767
768
769
770
771
772
773
774
775
776
777
778
779
780
781
782
783
784
785
786
787
788
789
790
791
792
793
794
795
796
797
798
799
800
801
802
803
804
805
806
807
808
809
810
811
812
813
814
815
816
817
818
819
820
821
822
823
824
825
826
827
828
829
830
831
832
833
834
835
836
837
838
839
840
841
842
843
844
845
846
847
848
849
850
851
852
853
854
855
856
857
858
859
860
861
862
863
864
865
866
867
868
869
870
871
872
873
874
875
876
877
878
879
880
881
882
883
884
885
886
887
888
889
890
891
892
893
894
895
896
897
898
899
900
901
902
903
904
905
906
907
908
909
910
911
912
913
914
915
916
917
918
919
920
921
922
923
924
925
926
927
928
929
930
931
932
933
934
935
936
937
938
939
940
941
942
943
944
945
946
947
948
949
950
951
952
953
954
955
956
957
958
959
960
961
962
963
964
965
966
967
968
969
970
971
972
973
974
975
976
977
978
979
980
981
982
983
984
985
986
987
988
989
990
991
992
993
994
995
996
997
998
999
1000
1001
1002
1003
1004
1005
1006
1007
1008
1009
1010
1011
1012
1013
1014
1015
1016
1017
1018
1019
1020
1021
1022
1023
1024
1025
1026
1027
1028
1029
1030
1031
1032
1033
1034
1035
1036
1037
1038
1039
1040
1041
1042
1043
1044
1045
1046
1047
1048
1049
1050
1051
1052
1053
1054
1055
1056
1057
1058
1059
1060
1061
1062
1063
1064
1065
1066
1067
1068
1069
1070
1071
1072
1073
1074
1075
1076
1077
1078
1079
1080
1081
1082
1083
1084
1085
1086
1087
1088
1089
1090
1091
1092
1093
1094
1095
1096
1097
1098
1099
1100
1101
1102
1103
1104
1105
1106
1107
1108
1109
1110
1111
1112
1113
1114
1115
1116
1117
1118
1119
1120
1121
1122
1123
1124
1125
1126
1127
1128
1129
1130
1131
1132
1133
1134
1135
1136
1137
1138
1139
1140
1141
1142
1143
1144
1145
1146
1147
1148
1149
1150
1151
1152
1153
1154
1155
1156
1157
1158
1159
1160
1161
1162
1163
1164
1165
1166
1167
1168
1169
1170
1171
1172
1173
1174
1175
1176
1177
1178
1179
1180
1181
1182
1183
1184
1185
1186
1187
1188
1189
1190
1191
1192
1193
1194
1195
1196
1197
1198
1199
1200
1201
1202
1203
1204
1205
1206
1207
1208
1209
1210
1211
1212
1213
1214
1215
1216
1217
1218
1219
1220
1221
1222
1223
1224
1225
1226
1227
1228
1229
1230
1231
1232
1233
1234
1235
1236
1237
1238
1239
1240
1241
1242
1243
1244
1245
1246
1247
1248
1249
1250
1251
1252
1253
1254
1255
1256
1257
1258
1259
1260
1261
1262
1263
1264
1265
1266
1267
1268
1269
1270
1271
1272
1273
1274
1275
1276
1277
1278
1279
1280
1281
1282
1283
1284
1285
1286
1287
1288
1289
1290
1291
1292
1293
1294
1295
1296
1297
1298
1299
1300
1301
1302
1303
1304
1305
1306
1307
1308
1309
1310
1311
1312
1313
1314
1315
1316
1317
1318
1319
1320
1321
1322
1323
1324
1325
1326
1327
1328
1329
1330
1331
1332
1333
1334
1335
1336
1337
1338
1339
1340
1341
1342
1343
1344
1345
1346
1347
1348
1349
1350
1351
1352
1353
1354
1355
1356
1357
1358
1359
1360
1361
1362
1363
1364
1365
1366
1367
1368
1369
1370
1371
1372
1373
1374
1375
1376
1377
1378
1379
1380
1381
1382
1383
1384
<?xml version="1.0" encoding="ISO8859-1" standalone="no"?>
<!DOCTYPE article PUBLIC "-//FreeBSD//DTD DocBook XML V4.2-Based Extension//EN"
	"../../../share/sgml/freebsd42.dtd" [
<!ENTITY % entities PUBLIC "-//FreeBSD//ENTITIES DocBook FreeBSD Entity Set//EN" "../../share/sgml/entities.ent">
%entities;
]>

<article lang='en'>
  <articleinfo>
    <title>Practical rc.d scripting in BSD</title>

    <author>
      <firstname>Yar</firstname>

      <surname>Tikhiy</surname>

      <affiliation>
	<address><email>yar@FreeBSD.org</email></address>
      </affiliation>
    </author>

    <copyright>
      <year>2005</year>
      <year>2006</year>
      <year>2012</year>

      <holder>The FreeBSD Project</holder>
    </copyright>

    <legalnotice id="trademarks" role="trademarks">
      &tm-attrib.freebsd;
      &tm-attrib.netbsd;
      &tm-attrib.general;
    </legalnotice>

    <pubdate>$FreeBSD$</pubdate>

    <releaseinfo>$FreeBSD$</releaseinfo>

    <abstract>
      <para>Beginners may find it difficult to relate the
	facts from the formal documentation on the BSD
	<filename>rc.d</filename> framework with the practical tasks
	of <filename>rc.d</filename> scripting.  In this article,
	we consider a few typical cases of increasing complexity,
	show <filename>rc.d</filename> features suited for each
	case, and discuss how they work.  Such an examination should
	provide reference points for further study of the design
	and efficient application of <filename>rc.d</filename>.</para>
    </abstract>
  </articleinfo>

  <sect1 id="rcng-intro">
    <title>Introduction</title>

    <para>The historical BSD had a monolithic startup script,
      <filename>/etc/rc</filename>.  It was invoked by
      &man.init.8; at system boot time and performed all userland
      tasks required for multi-user operation: checking and
      mounting file systems, setting up the network, starting
      daemons, and so on.  The precise list of tasks was not the
      same in every system; admins needed to customize it.  With
      few exceptions, <filename>/etc/rc</filename> had to be modified,
      and true hackers liked it.</para>

    <para>The real problem with the monolithic approach was that
      it provided no control over the individual components started
      from <filename>/etc/rc</filename>.  For instance,
      <filename>/etc/rc</filename> could not restart a single daemon.
      The system admin had to find the daemon process by hand, kill it,
      wait until it actually exited, then browse through
      <filename>/etc/rc</filename> for the flags, and finally type
      the full command line to start the daemon again.  The task
      would become even more difficult and prone to errors if the
      service to restart consisted of more than one daemon or
      demanded additional actions.  In a few words, the single
      script failed to fulfill what scripts are for: to make the
      system admin's life easier.</para>

    <para>Later there was an attempt to split out some parts of
      <filename>/etc/rc</filename> for the sake of starting the
      most important subsystems separately.  The notorious example
      was <filename>/etc/netstart</filename> to bring up networking.
      It did allow for accessing the network from single-user
      mode, but it did not integrate well into the automatic startup
      process because parts of its code needed to interleave with
      actions essentially unrelated to networking.  That was why
      <filename>/etc/netstart</filename> mutated into
      <filename>/etc/rc.network</filename>.  The latter was no
      longer an ordinary script; it comprised of large, tangled
      &man.sh.1; functions called from <filename>/etc/rc</filename>
      at different stages of system startup.  However, as the startup
      tasks grew diverse and sophisticated, the
      <quote>quasi-modular</quote> approach became even more of a
      drag than the monolithic <filename>/etc/rc</filename> had
      been.</para>

    <para>Without a clean and well-designed framework, the startup
      scripts had to bend over backwards to satisfy the needs of
      rapidly developing BSD-based operating systems.  It became
      obvious at last that more steps are necessary on the way to
      a fine-grained and extensible <filename>rc</filename> system.
      Thus BSD <filename>rc.d</filename> was born.  Its acknowledged
      fathers were Luke Mewburn and the NetBSD community.  Later
      it was imported into &os;.  Its name refers to the location
      of system scripts for individual services, which is in
      <filename>/etc/rc.d</filename>.  Soon we
      will learn about more components of the <filename>rc.d</filename>
      system and see how the individual scripts are invoked.</para>

    <para>The basic ideas behind BSD <filename>rc.d</filename> are
      <emphasis>fine modularity</emphasis> and <emphasis>code
      reuse</emphasis>.  <emphasis>Fine modularity</emphasis> means
      that each basic <quote>service</quote> such as a system daemon
      or primitive startup task gets its own &man.sh.1; script able
      to start the service, stop it, reload it, check its status.
      A particular action is chosen by the command-line argument
      to the script.  The <filename>/etc/rc</filename> script still
      drives system startup, but now it merely invokes the smaller
      scripts one by one with the <option>start</option> argument.
      It is easy to perform shutdown tasks as well by running the
      same set of scripts with the <option>stop</option> argument,
      which is done by <filename>/etc/rc.shutdown</filename>.  Note
      how closely this follows the Unix way of having a set of small
      specialized tools, each fulfilling its task as well as possible.
      <emphasis>Code reuse</emphasis> means that common operations
      are implemented as &man.sh.1; functions and collected in
      <filename>/etc/rc.subr</filename>.  Now a typical script can
      be just a few lines' worth of &man.sh.1; code.  Finally, an
      important part of the <filename>rc.d</filename> framework is
      &man.rcorder.8;, which helps <filename>/etc/rc</filename> to
      run the small scripts orderly with respect to dependencies
      between them.  It can help <filename>/etc/rc.shutdown</filename>,
      too, because the proper order for the shutdown sequence is
      opposite to that of startup.</para>

    <para>The BSD <filename>rc.d</filename> design is described in
      <link linkend="lukem">the original article by Luke Mewburn</link>,
      and the <filename>rc.d</filename> components are documented
      in great detail in <link linkend="manpages">the respective
      manual pages</link>.  However, it might not appear obvious
      to an <filename>rc.d</filename> newbie how to tie the numerous
      bits and pieces together in order to create a well-styled
      script for a particular task.  Therefore this article will
      try a different approach to describe <filename>rc.d</filename>.
      It will show which features should be used in a number of
      typical cases, and why.  Note that this is not a how-to
      document because our aim is not at giving ready-made recipes,
      but at showing a few easy entrances into the
      <filename>rc.d</filename> realm.  Neither is this article a
      replacement for the relevant manual pages.  Do not hesitate
      to refer to them for more formal and complete documentation
      while reading this article.</para>

    <para>There are prerequisites to understanding this article.
      First of all, you should be familiar with the &man.sh.1;
      scripting language in order to master <filename>rc.d</filename>.
      In addition, you should know how the system performs
      userland startup and shutdown tasks, which is described in
      &man.rc.8;.</para>

    <para>This article focuses on the &os; branch of
      <filename>rc.d</filename>.  Nevertheless, it may be useful
      to NetBSD developers, too, because the two branches of BSD
      <filename>rc.d</filename> not only share the same design
      but also stay similar in their aspects visible to script
      authors.</para>
  </sect1>

  <sect1 id="rcng-task">
    <title>Outlining the task</title>

    <para>A little consideration before starting
      <envar>$EDITOR</envar> will not hurt.  In order to write a
      well-tempered <filename>rc.d</filename> script for a system
      service, we should be able to answer the following questions
      first:</para>

    <itemizedlist>
      <listitem>
	<para>Is the service mandatory or optional?</para>
      </listitem>

      <listitem>
	<para>Will the script serve a single program, e.g.,
	  a daemon, or perform more complex actions?</para>
      </listitem>

      <listitem>
	<para>Which other services will our service depend on,
	  and vice versa?</para>
      </listitem>
    </itemizedlist>

    <para>From the examples that follow we will see why it is
      important to know the answers to these questions.</para>
  </sect1>

  <sect1 id="rcng-dummy">
    <title>A dummy script</title>

    <para>The following script just emits a message each time the
      system boots up:</para>

    <informalexample>
      <programlisting>#!/bin/sh<co id="rcng-dummy-shebang"/>

. /etc/rc.subr<co id="rcng-dummy-include"/>

name="dummy"<co id="rcng-dummy-name"/>
start_cmd="${name}_start"<co id="rcng-dummy-startcmd"/>
stop_cmd=":"<co id="rcng-dummy-stopcmd"/>

dummy_start()<co id="rcng-dummy-startfn"/>
{
	echo "Nothing started."
}

load_rc_config $name<co id="rcng-dummy-loadconfig"/>
run_rc_command "$1"<co id="rcng-dummy-runcommand"/></programlisting>
    </informalexample>

    <para>Things to note are:</para>

    <calloutlist>
      <callout arearefs="rcng-dummy-shebang">
	<para>An interpreted script should begin with the magic
	  <quote>shebang</quote> line.  That line specifies the
	  interpreter program for the script.  Due to the shebang
	  line, the script can be invoked exactly like a binary
	  program provided that it has the execute bit set.
	  (See &man.chmod.1;.)
	  For example, a system admin can run our script manually,
	  from the command line:</para>

	<screen>&prompt.root; <userinput>/etc/rc.d/dummy start</userinput></screen>

	<note>
	  <para>In order to be properly managed by the
	    <filename>rc.d</filename> framework, its scripts need
	    to be written in the &man.sh.1; language.  If you have
	    a service or port that uses a binary control utility
	    or a startup routine written in another language,
	    install that element in <filename>/usr/sbin</filename>
	    (for the system) or <filename>/usr/local/sbin</filename>
	    (for ports) and call it from a &man.sh.1; script in the
	    appropriate <filename>rc.d</filename> directory.</para>
	</note>

	<tip>
	  <para>If you would like to learn the details of why
	    <filename>rc.d</filename> scripts must be written in
	    the &man.sh.1; language, see how <filename>/etc/rc</filename>
	    invokes them by means of <function>run_rc_script</function>,
	    then study the implementation of
	    <function>run_rc_script</function> in
	    <filename>/etc/rc.subr</filename>.</para>
	</tip>
      </callout>

      <callout arearefs="rcng-dummy-include">
	<para>In <filename>/etc/rc.subr</filename>, a number of
	  &man.sh.1; functions are defined for an <filename>rc.d</filename>
	  script to use.  The functions are documented in
	  &man.rc.subr.8;.  While it is theoretically possible to
	  write an <filename>rc.d</filename> script without ever
	  using &man.rc.subr.8;, its functions prove extremely handy
	  and make the job an order of magnitude easier.  So it is
	  no surprise that everybody resorts to &man.rc.subr.8; in
	  <filename>rc.d</filename> scripts.  We are not going to
	  be an exception.</para>

	<para>An <filename>rc.d</filename> script must
	  <quote>source</quote> <filename>/etc/rc.subr</filename>
	  (include it using <quote><command>.</command></quote>)
	  <emphasis>before</emphasis> it calls &man.rc.subr.8;
	  functions so that &man.sh.1; has an opportunity to learn
	  the functions.  The preferred style is to source
	  <filename>/etc/rc.subr</filename> first of all.</para>

	<note>
	  <para>Some useful functions related to networking
	    are provided by another include file,
	    <filename>/etc/network.subr</filename>.</para>
	</note>
      </callout>

      <callout arearefs="rcng-dummy-name">
	<para><anchor id="name-var"/>The mandatory variable
	  <envar>name</envar> specifies the name of our script.  It
	  is required by &man.rc.subr.8;.  That is, each
	  <filename>rc.d</filename> script <emphasis>must</emphasis>
	  set <envar>name</envar> before it calls &man.rc.subr.8;
	  functions.</para>

	<para>Now it is the right time to choose a unique name for
	  our script once and for all.  We will use it in a number
	  of places while developing the script.  For a start, let
	  us give the same name to the script file, too.</para>

	<note>
	  <para>The current style of <filename>rc.d</filename>
	    scripting is to enclose values assigned to variables
	    in double quotes.  Keep in mind that it is just a style
	    issue that may not always be applicable.  You can
	    safely omit quotes from around simple words without
	    &man.sh.1; metacharacters in them, while in certain
	    cases you will need single quotes to prevent any
	    interpretation of the value by &man.sh.1;.  A programmer
	    should be able to tell the language syntax from style
	    conventions and use both of them wisely.</para>
	</note>
      </callout>

      <callout arearefs="rcng-dummy-startcmd">
	<para>The main idea behind &man.rc.subr.8; is that an
	  <filename>rc.d</filename> script provides handlers, or
	  methods, for &man.rc.subr.8; to invoke.  In particular,
	  <option>start</option>, <option>stop</option>, and other
	  arguments to an <filename>rc.d</filename> script are
	  handled this way.  A method is a &man.sh.1; expression
	  stored in a variable named
	  <envar><replaceable>argument</replaceable>_cmd</envar>,
	  where <replaceable>argument</replaceable> corresponds to
	  what can be specified on the script's command line.  We
	  will see later how &man.rc.subr.8; provides default methods
	  for the standard arguments.</para>

	<note>
	  <para>To make the code in <filename>rc.d</filename> more
	    uniform, it is common to use <envar>${name}</envar>
	    wherever appropriate.  Thus a number of lines can be just
	    copied from one script to another.</para>
	</note>
      </callout>

      <callout arearefs="rcng-dummy-stopcmd">
	<para>We should keep in mind that &man.rc.subr.8; provides
	  default methods for the standard arguments.  Consequently,
	  we must override a standard method with a no-op &man.sh.1;
	  expression if we want it to do nothing.</para>
      </callout>

      <callout arearefs="rcng-dummy-startfn">
	<para>The body of a sophisticated method can be implemented
	  as a function.  It is a good idea to make the function
	  name meaningful.</para>

	<important>
	  <para>It is strongly recommended to add the prefix
	    <envar>${name}</envar> to the names of all functions
	    defined in our script so they never clash with the
	    functions from &man.rc.subr.8; or another common include
	    file.</para>
	</important>
      </callout>

      <callout arearefs="rcng-dummy-loadconfig">
	<para>This call to &man.rc.subr.8; loads &man.rc.conf.5;
	  variables.  Our script makes no use of them yet, but it
	  still is recommended to load &man.rc.conf.5; because there
	  can be &man.rc.conf.5; variables controlling &man.rc.subr.8;
	  itself.</para>
      </callout>

      <callout arearefs="rcng-dummy-runcommand">
	<para>Usually this is the last command in an
	  <filename>rc.d</filename> script.  It invokes the
	  &man.rc.subr.8; machinery to perform the requested action
	  using the variables and methods our script has provided.</para>
      </callout>
    </calloutlist>
  </sect1>

  <sect1 id="rcng-confdummy">
    <title>A configurable dummy script</title>

    <para>Now let us add some controls to our dummy script.  As you
      may know, <filename>rc.d</filename> scripts are controlled
      with &man.rc.conf.5;.  Fortunately, &man.rc.subr.8; hides all
      the complications from us.  The following script uses
      &man.rc.conf.5; via &man.rc.subr.8; to see whether it is
      enabled in the first place, and to fetch a message to show
      at boot time.  These two tasks in fact are independent.  On
      the one hand, an <filename>rc.d</filename> script can just
      support enabling and disabling its service.  On the other
      hand, a mandatory <filename>rc.d</filename> script can have
      configuration variables.  We will do both things in the same
      script though:</para>

    <informalexample>
      <programlisting>#!/bin/sh

. /etc/rc.subr

name=dummy
rcvar=dummy_enable<co id="rcng-confdummy-rcvar"/>

start_cmd="${name}_start"
stop_cmd=":"

load_rc_config $name<co id="rcng-confdummy-loadconfig"/>
eval "${rcvar}=\${${rcvar}:-'NO'}"<co id="rcng-confdummy-enable"/>
dummy_msg=${dummy_msg:-"Nothing started."}<co id="rcng-confdummy-opt"/>

dummy_start()
{
	echo "$dummy_msg"<co id="rcng-confdummy-msg"/>
}

run_rc_command "$1"</programlisting>
    </informalexample>

    <para>What changed in this example?</para>

    <calloutlist>
      <callout arearefs="rcng-confdummy-rcvar">
	<para>The variable <envar>rcvar</envar> specifies
	  the name of the ON/OFF knob variable.</para>
      </callout>

      <callout arearefs="rcng-confdummy-loadconfig">
	<para>Now <function>load_rc_config</function> is invoked
	  earlier in the script, before any &man.rc.conf.5; variables
	  are accessed.</para>

	<note>
	  <para>While examining <filename>rc.d</filename> scripts,
	    keep in mind that &man.sh.1; defers the evaluation of
	    expressions in a function until the latter is called.
	    Therefore it is not an error to invoke
	    <function>load_rc_config</function> as late as just
	    before <function>run_rc_command</function> and still
	    access &man.rc.conf.5; variables from the method functions
	    exported to <function>run_rc_command</function>.  This
	    is because the method functions are to be called by
	    <function>run_rc_command</function>, which is invoked
	    <emphasis>after</emphasis>
	    <function>load_rc_config</function>.</para>
	</note>
      </callout>

      <callout arearefs="rcng-confdummy-enable">
	<para>A warning will be emitted by
	  <function>run_rc_command</function> if <envar>rcvar</envar>
	  itself is set, but the indicated knob variable is unset.
	  If your <filename>rc.d</filename> script is for the base
	  system, you should add a default setting for the knob to
	  <filename>/etc/defaults/rc.conf</filename> and document
	  it in &man.rc.conf.5;.  Otherwise it is your script that
	  should provide a default setting for the knob.  A portable
	  approach to the latter case is shown in the example.</para>

	<note>
	  <para>You can make &man.rc.subr.8; act as though the knob
	    is set to <literal>ON</literal>, irrespective of its
	    current setting, by prefixing the argument to the script
	    with <literal>one</literal> or <literal>force</literal>,
	    as in <option>onestart</option> or <option>forcestop</option>.
	    Keep in mind though that <literal>force</literal> has
	    other dangerous effects we will touch upon below, while
	    <literal>one</literal> just overrides the ON/OFF knob.
	    E.g., assume that <envar>dummy_enable</envar> is
	    <literal>OFF</literal>.  The following command will run
	    the <option>start</option> method in spite of the
	    setting:</para>

	  <screen>&prompt.root; <userinput>/etc/rc.d/dummy onestart</userinput></screen>
	</note>
      </callout>

      <callout arearefs="rcng-confdummy-opt">
	<para>Now the message to be shown at boot time is no
	  longer hard-coded in the script.  It is specified by an
	  &man.rc.conf.5; variable named <envar>dummy_msg</envar>.
	  This is a trivial example of how &man.rc.conf.5; variables
	  can control an <filename>rc.d</filename> script.</para>

	<important>
	  <para>The names of all &man.rc.conf.5; variables used
	    exclusively by our script <emphasis>must</emphasis>
	    have the same prefix: <envar>${name}</envar>.  For
	    example: <envar>dummy_mode</envar>,
	    <envar>dummy_state_file</envar>, and so on.</para>
	</important>

	<note>
	  <para>While it is possible to use a shorter name internally,
	    e.g., just <envar>msg</envar>, adding the unique prefix
	    <envar>${name}</envar> to all global names introduced by
	    our script will save us from possible
	    collisions with the &man.rc.subr.8; namespace.</para>

	  <para>As long as an &man.rc.conf.5; variable and its
	    internal equivalent are the same, we can use a more
	    compact expression to set the default value:</para>

	  <programlisting>: ${dummy_msg:="Nothing started."}</programlisting>

	  <para>The current style is to use the more verbose form
	    though.</para>

	  <para>As a rule, <filename>rc.d</filename> scripts of the
	    base system need not provide defaults for their
	    &man.rc.conf.5; variables because the defaults should
	    be set in <filename>/etc/defaults/rc.conf</filename>
	    instead.  On the other hand, <filename>rc.d</filename>
	    scripts for ports should provide the defaults as shown
	    in the example.</para>
	</note>
      </callout>

      <callout arearefs="rcng-confdummy-msg">
	<para>Here we use <envar>dummy_msg</envar> to actually
	  control our script, i.e., to emit a variable message.</para>
      </callout>
    </calloutlist>
  </sect1>

  <sect1 id="rcng-daemon">
    <title>Startup and shutdown of a simple daemon</title>

    <para>We said earlier that &man.rc.subr.8; could provide default
      methods.  Obviously, such defaults cannot be too general.
      They are suited for the common case of starting and shutting
      down a simple daemon program.  Let us assume now that we need
      to write an <filename>rc.d</filename> script for such a daemon
      called <command>mumbled</command>.  Here it is:</para>

    <informalexample>
      <programlisting>#!/bin/sh

. /etc/rc.subr

name=mumbled
rcvar=mumbled_enable

command="/usr/sbin/${name}"<co id="rcng-daemon-basic-cmd"/>

load_rc_config $name
run_rc_command "$1"</programlisting>
    </informalexample>

    <para>Pleasingly simple, isn't it?  Let us examine our little
      script.  The only new thing to note is as follows:</para>

    <calloutlist>
      <callout arearefs="rcng-daemon-basic-cmd">
	<para>The <envar>command</envar> variable is meaningful to
	  &man.rc.subr.8;.  If it is set, &man.rc.subr.8; will act
	  according to the scenario of serving a conventional daemon.
	  In particular, the default methods will be provided for
	  such arguments: <option>start</option>, <option>stop</option>,
	  <option>restart</option>, <option>poll</option>, and
	  <option>status</option>.</para>

	<para>The daemon will be started by running
	  <envar>$command</envar> with command-line flags specified
	  by <envar>$mumbled_flags</envar>.  Thus all the input
	  data for the default <option>start</option> method are
	  available in the variables set by our script.  Unlike
	  <option>start</option>, other methods may require additional
	  information about the process started.  For instance,
	  <option>stop</option> must know the PID of the process
	  to terminate it.  In the present case, &man.rc.subr.8;
	  will scan through the list of all processes, looking for
	  a process with its name equal to <envar>$procname</envar>.
	  The latter is another variable of meaning to &man.rc.subr.8;,
	  and its value defaults to that of <envar>command</envar>.
	  In other words, when we set <envar>command</envar>,
	  <envar>procname</envar> is effectively set to the same
	  value.  This enables our script to kill the daemon and
	  to check if it is running in the first place.</para>

	<note>
	  <para>Some programs are in fact executable scripts.  The
	    system runs such a script by starting its interpreter
	    and passing the name of the script to it as a command-line
	    argument.  This is reflected in the list of processes,
	    which can confuse &man.rc.subr.8;.  You should additionally
	    set <envar>command_interpreter</envar> to let &man.rc.subr.8;
	    know the actual name of the process if <envar>$command</envar>
	    is a script.</para>

	  <para>For each <filename>rc.d</filename> script, there
	    is an optional &man.rc.conf.5; variable that takes
	    precedence over <envar>command</envar>.  Its name is
	    constructed as follows: <envar>${name}_program</envar>,
	    where <envar>name</envar> is the mandatory variable we
	    discussed <link linkend="name-var">earlier</link>.
	    E.g., in this case it will be <envar>mumbled_program</envar>.
	    It is &man.rc.subr.8; that arranges
	    <envar>${name}_program</envar> to override
	    <envar>command</envar>.</para>

	  <para>Of course, &man.sh.1; will permit you to set
	    <envar>${name}_program</envar> from &man.rc.conf.5; or
	    the script itself even if <envar>command</envar> is
	    unset.  In that case, the special properties of
	    <envar>${name}_program</envar> are lost, and it becomes
	    an ordinary variable your script can use for its own
	    purposes.  However, the sole use of
	    <envar>${name}_program</envar> is discouraged because
	    using it together with <envar>command</envar> became
	    an idiom of <filename>rc.d</filename> scripting.</para>
	</note>

	<para>For more detailed information on default methods,
	  refer to &man.rc.subr.8;.</para>
      </callout>
    </calloutlist>
  </sect1>

  <sect1 id="rcng-daemon-adv">
    <title>Startup and shutdown of an advanced daemon</title>

    <para>Let us add some meat onto the bones of the previous
      script and make it more complex and featureful.  The default
      methods can do a good job for us, but we may need some of
      their aspects tweaked.  Now we will learn how to tune the
      default methods to our needs.</para>

    <informalexample>
      <programlisting>#!/bin/sh

. /etc/rc.subr

name=mumbled
rcvar=mumbled_enable

command="/usr/sbin/${name}"
command_args="mock arguments &gt; /dev/null 2&gt;&amp;1"<co id="rcng-daemon-adv-args"/>

pidfile="/var/run/${name}.pid"<co id="rcng-daemon-adv-pid"/>

required_files="/etc/${name}.conf /usr/share/misc/${name}.rules"<co id="rcng-daemon-adv-reqfiles"/>

sig_reload="USR1"<co id="rcng-daemon-adv-sig"/>

start_precmd="${name}_prestart"<co id="rcng-daemon-adv-precmd"/>
stop_postcmd="echo Bye-bye"<co id="rcng-daemon-adv-postcmd"/>

extra_commands="reload plugh xyzzy"<co id="rcng-daemon-adv-extra"/>

plugh_cmd="mumbled_plugh"<co id="rcng-daemon-adv-methods"/>
xyzzy_cmd="echo 'Nothing happens.'"

mumbled_prestart()
{
	if checkyesno mumbled_smart; then<co id="rcng-daemon-adv-yn"/>
		rc_flags="-o smart ${rc_flags}"<co id="rcng-daemon-adv-rcflags"/>
	fi
	case "$mumbled_mode" in
	foo)
		rc_flags="-frotz ${rc_flags}"
		;;
	bar)
		rc_flags="-baz ${rc_flags}"
		;;
	*)
		warn "Invalid value for mumbled_mode"<co id="rcng-daemon-adv-warn"/>
		return 1<co id="rcng-daemon-adv-preret"/>
		;;
	esac
	run_rc_command xyzzy<co id="rcng-daemon-adv-run"/>
	return 0
}

mumbled_plugh()<co id="rcng-daemon-adv-plugh"/>
{
	echo 'A hollow voice says "plugh".'
}

load_rc_config $name
run_rc_command "$1"</programlisting>
    </informalexample>

    <calloutlist>
      <callout arearefs="rcng-daemon-adv-args">
	<para>Additional arguments to <envar>$command</envar> can
	  be passed in <envar>command_args</envar>.  They will be
	  added to the command line after <envar>$mumbled_flags</envar>.
	  Since the final command line is passed to <command>eval</command>
	  for its actual execution, input and output redirections
	  can be specified in <envar>command_args</envar>.</para>

	<note>
	  <para><emphasis>Never</emphasis> include dashed options,
	    like <option>-X</option> or <option>--foo</option>, in
	    <envar>command_args</envar>.
	    The contents of <envar>command_args</envar> will
	    appear at the end of the final command line, hence
	    they are likely to follow arguments present in
	    <envar>${name}_flags</envar>; but most commands will
	    not recognize dashed options after ordinary arguments.
	    A better way of passing additional options
	    to <envar>$command</envar> is to add them
	    to the beginning of <envar>${name}_flags</envar>.
	    Another way is to modify <envar>rc_flags</envar> <link
	    linkend="rc-flags">as shown later</link>.</para>
	</note>
      </callout>

      <callout arearefs="rcng-daemon-adv-pid">
	<para>A good-mannered daemon should create a
	  <emphasis>pidfile</emphasis> so that its process can be
	  found more easily and reliably.  The variable
	  <envar>pidfile</envar>, if set, tells &man.rc.subr.8;
	  where it can find the pidfile for its default methods to
	  use.</para>

	<note>
	  <para>In fact, &man.rc.subr.8; will also use the pidfile
	    to see if the daemon is already running before starting
	    it.  This check can be skipped by using the
	    <option>faststart</option> argument.</para>
	</note>
      </callout>

      <callout arearefs="rcng-daemon-adv-reqfiles">
	<para>If the daemon cannot run unless certain files exist,
	  just list them in <envar>required_files</envar>, and
	  &man.rc.subr.8; will check that those files do exist
	  before starting the daemon.  There also are
	  <envar>required_dirs</envar> and <envar>required_vars</envar>
	  for directories and environment variables, respectively.
	  They all are described in detail in &man.rc.subr.8;.</para>

	<note>
	  <para>The default method from &man.rc.subr.8; can be
	    forced to skip the prerequisite checks by using
	    <option>forcestart</option> as the argument to the
	    script.</para>
	</note>
      </callout>

      <callout arearefs="rcng-daemon-adv-sig">
	<para>We can customize signals to send to the daemon in
	  case they differ from the well-known ones.  In particular,
	  <envar>sig_reload</envar> specifies the signal that makes
	  the daemon reload its configuration; it is
	  <symbol>SIGHUP</symbol> by default.  Another signal is
	  sent to stop the daemon process; the default is
	  <symbol>SIGTERM</symbol>, but this can be changed by
	  setting <envar>sig_stop</envar> appropriately.</para>

	<note>
	  <para>The signal names should be specified to &man.rc.subr.8;
	    without the <literal>SIG</literal> prefix, as it is
	    shown in the example.  The &os; version of &man.kill.1;
	    can recognize the <literal>SIG</literal> prefix, but
	    the versions from other OS types may not.</para>
	</note>
      </callout>

      <callout arearefs="rcng-daemon-adv-precmd rcng-daemon-adv-postcmd">
	<para>Performing additional tasks before or after the default
	  methods is easy.  For each command-argument supported by
	  our script, we can define
	  <envar><replaceable>argument</replaceable>_precmd</envar> and
	  <envar><replaceable>argument</replaceable>_postcmd</envar>.
	  These &man.sh.1; commands are invoked before and after
	  the respective method, as it is evident from their
	  names.</para>

	<note>
	  <para>Overriding a default method with a custom
	    <envar><replaceable>argument</replaceable>_cmd</envar>
	    still does not prevent us from making use of
	    <envar><replaceable>argument</replaceable>_precmd</envar> or
	    <envar><replaceable>argument</replaceable>_postcmd</envar>
	    if we need to.  In particular, the former is good for
	    checking custom, sophisticated conditions that should
	    be met before performing the command itself.  Using
	    <envar><replaceable>argument</replaceable>_precmd</envar> along
	    with <envar><replaceable>argument</replaceable>_cmd</envar>
	    lets us logically separate the checks from the
	    action.</para>

	  <para>Do not forget that you can cram any valid &man.sh.1;
	    expressions into the methods, pre-, and post-commands
	    you define.  Just invoking a function that makes the
	    real job is a good style in most cases, but never let
	    style limit your understanding of what is going on
	    behind the curtain.</para>
	</note>
      </callout>

      <callout arearefs="rcng-daemon-adv-extra">
	<para>If we would like to implement custom arguments, which
	  can also be thought of as <emphasis>commands</emphasis>
	  to our script, we need to list them in
	  <envar>extra_commands</envar> and provide methods to
	  handle them.</para>

	<note>
	  <para>The <option>reload</option> command is special.  On
	    the one hand, it has a preset method in &man.rc.subr.8;.
	    On the other hand, <option>reload</option> is not offered
	    by default.  The reason is that not all daemons use the
	    same reload mechanism and some have nothing to reload
	    at all.  So we need to ask explicitly that the builtin
	    functionality be provided.  We can do so via
	    <envar>extra_commands</envar>.</para>

	  <para>What do we get from the default method for
	    <option>reload</option>?  Quite often daemons reload
	    their configuration upon reception of a signal &mdash;
	    typically, <symbol>SIGHUP</symbol>.  Therefore
	    &man.rc.subr.8; attempts to reload the daemon by sending
	    a signal to it.  The signal is preset to
	    <symbol>SIGHUP</symbol> but can be customized via
	    <envar>sig_reload</envar> if necessary.</para>
	</note>
      </callout>

      <callout arearefs="rcng-daemon-adv-methods rcng-daemon-adv-plugh">
	<para>Our script supports two non-standard commands,
	  <option>plugh</option> and <option>xyzzy</option>.  We
	  saw them listed in <envar>extra_commands</envar>, and now
	  it is time to provide methods for them.  The method for
	  <option>xyzzy</option> is just inlined while that for
	  <option>plugh</option> is implemented as the
	  <function>mumbled_plugh</function> function.</para>

	<para>Non-standard commands are not invoked during startup
	  or shutdown.  Usually they are for the system admin's
	  convenience.  They can also be used from other subsystems,
	  e.g., &man.devd.8; if specified in &man.devd.conf.5;.</para>

	<para>The full list of available commands can be found in
	  the usage line printed by &man.rc.subr.8; when the script
	  is invoked without arguments.  For example, here is the
	  usage line from the script under study:</para>

	<screen>&prompt.root; <userinput>/etc/rc.d/mumbled</userinput>
Usage: /etc/rc.d/mumbled [fast|force|one](start|stop|restart|rcvar|reload|plugh|xyzzy|status|poll)</screen>
      </callout>

      <callout arearefs="rcng-daemon-adv-run">
	<para>A script can invoke its own standard or non-standard
	  commands if needed.  This may look similar to calling
	  functions, but we know that commands and shell functions
	  are not always the same thing.  For instance,
	  <command>xyzzy</command> is not implemented as a function
	  here.  In addition, there can be a pre-command and
	  post-command, which should be invoked orderly.  So the
	  proper way for a script to run its own command is by means
	  of &man.rc.subr.8;, as shown in the example.</para>
      </callout>

      <callout arearefs="rcng-daemon-adv-yn">
	<para>A handy function named <function>checkyesno</function>
	  is provided by &man.rc.subr.8;.  It takes a variable name
	  as its argument and returns a zero exit code if and only
	  if the variable is set to <literal>YES</literal>, or
	  <literal>TRUE</literal>, or <literal>ON</literal>, or
	  <literal>1</literal>, case insensitive; a non-zero exit
	  code is returned otherwise.  In the latter case, the
	  function tests the variable for being set to
	  <literal>NO</literal>, <literal>FALSE</literal>,
	  <literal>OFF</literal>, or <literal>0</literal>, case
	  insensitive; it prints a warning message if the variable
	  contains anything else, i.e., junk.</para>

	<para>Keep in mind that for &man.sh.1; a zero exit code
	  means true and a non-zero exit code means false.</para>

	<important>
	  <para>The <function>checkyesno</function> function takes
	    a <emphasis>variable name</emphasis>.  Do not pass the
	    expanded <emphasis>value</emphasis> of a variable to
	    it; it will not work as expected.</para>

	  <para>The following is the correct usage of
	    <function>checkyesno</function>:</para>

	  <programlisting>if checkyesno mumbled_enable; then
        foo
fi</programlisting>

	  <para>On the contrary, calling <function>checkyesno</function>
	    as shown below will not work &mdash; at least not as
	    expected:</para>

	  <programlisting>if checkyesno "${mumbled_enable}"; then
        foo
fi</programlisting>
	</important>
      </callout>

      <callout arearefs="rcng-daemon-adv-rcflags">
	<para><anchor id="rc-flags"/>We can affect the flags to be
	  passed to <envar>$command</envar> by modifying
	  <envar>rc_flags</envar> in <envar>$start_precmd</envar>.</para>
      </callout>

      <callout arearefs="rcng-daemon-adv-warn">
	<para>In certain cases we may need to emit an important
	  message that should go to <application>syslog</application>
	  as well.  This can be done easily with the following
	  &man.rc.subr.8; functions: <function>debug</function>,
	  <function>info</function>, <function>warn</function>, and
	  <function>err</function>.  The latter function then exits
	  the script with the code specified.</para>
      </callout>

      <callout arearefs="rcng-daemon-adv-preret">
	<para>The exit codes from methods and their pre-commands
	  are not just ignored by default.  If
	  <envar><replaceable>argument</replaceable>_precmd</envar> returns
	  a non-zero exit code, the main method will not be performed.
	  In turn,
	  <envar><replaceable>argument</replaceable>_postcmd</envar> will
	  not be invoked unless the main method returns a zero exit
	  code.</para>

	<note>
	  <para>However, &man.rc.subr.8; can be instructed from the
	    command line to ignore those exit codes and invoke all
	    commands anyway by prefixing an argument with
	    <literal>force</literal>, as in
	    <option>forcestart</option>.</para>
	</note>
      </callout>
    </calloutlist>
  </sect1>

  <sect1 id="rcng-hookup">
    <title>Connecting a script to the rc.d framework</title>

    <para>After a script has been written, it needs to be integrated
      into <filename>rc.d</filename>.  The crucial step is to install
      the script in <filename>/etc/rc.d</filename> (for the base
      system) or <filename>/usr/local/etc/rc.d</filename> (for
      ports).  Both &lt;<filename>bsd.prog.mk</filename>&gt; and
      &lt;<filename>bsd.port.mk</filename>&gt; provide convenient
      hooks for that, and usually you do not have to worry about
      the proper ownership and mode.  System scripts should be
      installed from <filename>src/etc/rc.d</filename> through the
      <filename>Makefile</filename> found there.  Port scripts can
      be installed using <makevar>USE_RC_SUBR</makevar> as described
      <ulink url="&url.books.porters-handbook;/rc-scripts.html">in
      the Porter's Handbook</ulink>.</para>

    <para>However, we should consider beforehand the place of
      our script in the system startup sequence.  The service handled
      by our script is likely to depend on other services.  For
      instance, a network daemon cannot function without the network
      interfaces and routing up and running.  Even if a service
      seems to demand nothing, it can hardly start before the basic
      filesystems have been checked and mounted.</para>

    <para>We mentioned &man.rcorder.8; already.  Now it is time to
      have a close look at it.  In a nutshell, &man.rcorder.8; takes
      a set of files, examines their contents, and prints a
      dependency-ordered list of files from the set to
      <varname>stdout</varname>.  The point is to keep dependency
      information <emphasis>inside</emphasis> the files so that
      each file can speak for itself only.  A file can specify the
      following information:</para>

    <itemizedlist>
      <listitem>
	<para>the names of the <quote>conditions</quote> (which means
	  services to us) it <emphasis>provides</emphasis>;</para>
      </listitem>

      <listitem>
	<para>the names of the <quote>conditions</quote>
	  it <emphasis>requires</emphasis>;</para>
      </listitem>

      <listitem>
	<para>the names of the <quote>conditions</quote> this file
	  should run <emphasis>before</emphasis>;</para>
      </listitem>

      <listitem>
	<para>additional <emphasis>keywords</emphasis> that can be
	  used to select a subset from the whole set of files
	  (&man.rcorder.8; can be instructed via options to include
	  or omit the files having particular keywords listed.)</para>
      </listitem>
    </itemizedlist>

    <para>It is no surprise that &man.rcorder.8; can handle only
      text files with a syntax close to that of &man.sh.1;.  That
      is, special lines understood by &man.rcorder.8; look like
      &man.sh.1; comments.  The syntax of such special lines is
      rather rigid to simplify their processing.  See &man.rcorder.8;
      for details.</para>

    <para>Besides using &man.rcorder.8; special lines, a script can
      insist on its dependency upon another service by just starting
      it forcibly.  This can be needed when the other service is
      optional and will not start by itself because the system admin
      has disabled it mistakenly in &man.rc.conf.5;.</para>

    <para>With this general knowledge in mind, let us consider the
      simple daemon script enhanced with dependency stuff:</para>

    <informalexample>
      <programlisting>#!/bin/sh

# PROVIDE: mumbled oldmumble <co id="rcng-hookup-provide"/>
# REQUIRE: DAEMON cleanvar frotz<co id="rcng-hookup-require"/>
# BEFORE:  LOGIN<co id="rcng-hookup-before"/>
# KEYWORD: nojail shutdown<co id="rcng-hookup-keyword"/>

. /etc/rc.subr

name=mumbled
rcvar=mumbled_enable

command="/usr/sbin/${name}"
start_precmd="${name}_prestart"

mumbled_prestart()
{
	if ! checkyesno frotz_enable &amp;&amp; \
	    ! /etc/rc.d/frotz forcestatus 1>/dev/null 2>&amp;1; then
		force_depend frotz || return 1<co id="rcng-hookup-force"/>
	fi
	return 0
}

load_rc_config $name
run_rc_command "$1"</programlisting>
    </informalexample>

    <para>As before, detailed analysis follows:</para>

    <calloutlist>
      <callout arearefs="rcng-hookup-provide">
	<para>That line declares the names of <quote>conditions</quote>
	  our script provides.  Now other scripts can record a
	  dependency on our script by those names.</para>

	<note>
	  <para>Usually a script specifies a single condition
	    provided.  However, nothing prevents us from listing
	    several conditions there, e.g., for compatibility
	    reasons.</para>

	  <para>In any case, the name of the main, or the only,
	    <literal>PROVIDE:</literal> condition should be the
	    same as <envar>${name}</envar>.</para>
	</note>
      </callout>

      <callout arearefs="rcng-hookup-require rcng-hookup-before">
	<para>So our script indicates which <quote>conditions</quote>
	  provided by other scripts it depends on.  According to
	  the lines, our script asks &man.rcorder.8; to put it after
	  the script(s) providing <filename>DAEMON</filename> and
	  <filename>cleanvar</filename>, but before that providing
	  <filename>LOGIN</filename>.</para>

	<note>
	  <para>The <literal>BEFORE:</literal> line should not be
	    abused to work around an incomplete dependency list in
	    the other script.  The appropriate case for using
	    <literal>BEFORE:</literal> is when the other script
	    does not care about ours, but our script can do its
	    task better if run before the other one.  A typical
	    real-life example is the network interfaces vs. the
	    firewall: While the interfaces do not depend on the
	    firewall in doing their job, the system security will
	    benefit from the firewall being ready before there is
	    any network traffic.</para>

	  <para>Besides conditions corresponding to a single service
	    each, there are meta-conditions and their
	    <quote>placeholder</quote> scripts used to ensure that
	    certain groups of operations are performed before others.
	    These are denoted by
	    <filename><replaceable>UPPERCASE</replaceable></filename>
	    names.  Their list and purposes can be found in
	    &man.rc.8;.</para>

	  <para>Keep in mind that putting a service name in the
	    <literal>REQUIRE:</literal> line does not guarantee
	    that the service will actually be running by the time
	    our script starts.  The required service may fail to
	    start or just be disabled in &man.rc.conf.5;.  Obviously,
	    &man.rcorder.8; cannot track such details, and &man.rc.8;
	    will not do that either.  Consequently, the application
	    started by our script should be able to cope with any
	    required services being unavailable.  In certain cases,
	    we can help it as discussed <link
	    linkend="forcedep">below.</link></para>
	</note>
      </callout>

      <callout arearefs="rcng-hookup-keyword">
	<para><anchor id="keywords"/>As we remember from the above text,
	  &man.rcorder.8; keywords can be used to select or leave
	  out some scripts.  Namely any &man.rcorder.8; consumer
	  can specify through <option>-k</option> and <option>-s</option>
	  options which keywords are on the <quote>keep list</quote> and
	  <quote>skip list</quote>, respectively.  From all the
	  files to be dependency sorted, &man.rcorder.8; will pick
	  only those having a keyword from the keep list (unless empty)
	  and not having a keyword from the skip list.</para>

	<para>In &os;, &man.rcorder.8; is used by
	  <filename>/etc/rc</filename> and
	  <filename>/etc/rc.shutdown</filename>.  These two scripts
	  define the standard list of &os; <filename>rc.d</filename>
	  keywords and their meanings as follows:</para>

	<variablelist>
	  <varlistentry>
	    <term><literal>nojail</literal></term>

	    <listitem>
	      <para>The service is not for &man.jail.8; environment.
		The automatic startup and shutdown procedures will
		ignore the script if inside a jail.</para>
	    </listitem>
	  </varlistentry>

	  <varlistentry>
	    <term><literal>nostart</literal></term>

	    <listitem>
	      <para>The service is to be started manually or not
		started at all.  The automatic startup procedure
		will ignore the script.  In conjunction with the
		<literal>shutdown</literal> keyword, this can be
		used to write scripts that do something only at
		system shutdown.</para>
	    </listitem>
	  </varlistentry>

	  <varlistentry>
	    <term><literal>shutdown</literal></term>

	    <listitem>
	      <para>This keyword is to be listed
		<emphasis>explicitly</emphasis> if the service needs
		to be stopped before system shutdown.</para>

	      <note>
		<para>When the system is going to shut down,
		  <filename>/etc/rc.shutdown</filename> runs.  It
		  assumes that most <filename>rc.d</filename> scripts
		  have nothing to do at that time.  Therefore
		  <filename>/etc/rc.shutdown</filename>
		  selectively invokes <filename>rc.d</filename>
		  scripts with the <literal>shutdown</literal>
		  keyword, effectively ignoring the
		  rest of the scripts.  For even faster shutdown,
		  <filename>/etc/rc.shutdown</filename> passes
		  the <option>faststop</option> command to the scripts
		  it runs so that they skip preliminary checks, e.g.,
		  the pidfile check.  As dependent services should be
		  stopped before their prerequisites,
		  <filename>/etc/rc.shutdown</filename> runs the scripts
		  in reverse dependency order.</para>

		<para>If writing a real
		  <filename>rc.d</filename> script, you should
		  consider whether it is relevant at system shutdown
		  time.  E.g., if your script does its work in
		  response to the <option>start</option> command
		  only, then you need not include this keyword.
		  However, if your script manages a service, it is
		  probably a good idea to stop it before the system
		  proceeds to the final stage of its shutdown
		  sequence described in &man.halt.8;.  In particular,
		  a service should be stopped explicitly if it needs
		  considerable time or special actions to shut down
		  cleanly.  A typical example of such a service is
		  a database engine.</para>
	      </note>
	    </listitem>
	  </varlistentry>
	</variablelist>
      </callout>

      <callout arearefs="rcng-hookup-force">
	<para><anchor id="forcedep"/>To begin with,
	  <function>force_depend</function> should be used with
	  much care.  It is generally better to revise the hierarchy
	  of configuration variables for your <filename>rc.d</filename>
	  scripts if they are interdependent.</para>

	<para>If you still cannot do without
	  <function>force_depend</function>, the example offers an
	  idiom of how to invoke it conditionally.  In the example,
	  our <command>mumbled</command> daemon requires that another
	  one, <command>frotz</command>, be started in advance.
	  However, <command>frotz</command> is optional, too; and
	  &man.rcorder.8; knows nothing about such details.
	  Fortunately, our script has access to all &man.rc.conf.5;
	  variables.  If <envar>frotz_enable</envar> is true, we
	  hope for the best and rely on <filename>rc.d</filename>
	  to have started <command>frotz</command>.  Otherwise we
	  forcibly check the status of <command>frotz</command>.
	  Finally, we enforce our dependency on <command>frotz</command>
	  if it is found to be not running.  A warning message will
	  be emitted by <function>force_depend</function> because
	  it should be invoked only if a misconfiguration has been
	  detected.</para>
      </callout>
    </calloutlist>
  </sect1>

  <sect1 id="rcng-args">
    <title>Giving more flexibility to an rc.d script</title>

    <para>When invoked during startup or shutdown, an
      <filename>rc.d</filename> script is supposed to act on the
      entire subsystem it is responsible for.  E.g.,
      <filename>/etc/rc.d/netif</filename> should start or stop all
      network interfaces described by &man.rc.conf.5;.  Either task
      can be uniquely indicated by a single command argument such
      as <option>start</option> or <option>stop</option>.  Between
      startup and shutdown, <filename>rc.d</filename> scripts help
      the admin to control the running system, and it is when the
      need for more flexibility and precision arises.  For instance,
      the admin may want to add the settings of a new network
      interface to &man.rc.conf.5; and then to start it without
      interfering with the operation of the existing interfaces.
      Next time the admin may need to shut down a single network
      interface.  In the spirit of the command line, the respective
      <filename>rc.d</filename> script calls for an extra argument,
      the interface name.</para>

    <para>Fortunately, &man.rc.subr.8; allows for passing any number
      of arguments to script's methods (within the system limits).
      Due to that, the changes in the script itself can be minimal.</para>
    
    <para>How can &man.rc.subr.8; gain
      access to the extra command-line arguments.  Should it just
      grab them directly?  Not by any means.  Firstly, an &man.sh.1;
      function has no access to the positional parameters of
      its caller, but &man.rc.subr.8; is just a sack of such
      functions.  Secondly, the good manner of
      <filename>rc.d</filename> dictates that it is for the
      main script to decide which arguments are to be passed
      to its methods.</para>

    <para>So the approach adopted by &man.rc.subr.8; is as follows:
      <function>run_rc_command</function> passes on all its
      arguments but the first one to the respective method verbatim.
      The first, omitted, argument is the name of the method itself:
      <option>start</option>, <option>stop</option>, etc.  It will
      be shifted out by <function>run_rc_command</function>,
      so what is <envar>$2</envar> in the original command line will
      be presented as <envar>$1</envar> to the method, and so on.</para>

    <para>To illustrate this opportunity, let us modify the primitive
      dummy script so that its messages depend on the additional
      arguments supplied.  Here we go:</para>

    <informalexample>
      <programlisting>#!/bin/sh

. /etc/rc.subr

name="dummy"
start_cmd="${name}_start"
stop_cmd=":"
kiss_cmd="${name}_kiss"
extra_commands="kiss"

dummy_start()
{
        if [ $# -gt 0 ]; then<co id="rcng-args-start"/>
                echo "Greeting message: $*"
        else
                echo "Nothing started."
        fi
}

dummy_kiss()
{
        echo -n "A ghost gives you a kiss"
        if [ $# -gt 0 ]; then<co id="rcng-args-kiss"/>
                echo -n " and whispers: $*"
        fi
        case "$*" in
        *[.!?])
                echo
                ;;
        *)
                echo .
                ;;
        esac
}

load_rc_config $name
run_rc_command "$@"<co id="rcng-args-all"/></programlisting>
    </informalexample>

    <para>What essential changes can we notice in the script?</para>

    <calloutlist>
      <callout arearefs="rcng-args-start">
	<para>All arguments you type after <option>start</option>
	  can end up as positional parameters to the respective
	  method.  We can use them in any way according to our
	  task, skills, and fancy.  In the current example, we just
	  pass all of them to &man.echo.1; as one string in the
	  next line &mdash; note <envar>$*</envar> within the double
	  quotes.  Here is how the script can be invoked now:</para>

	<screen>&prompt.root; <userinput>/etc/rc.d/dummy start</userinput>
Nothing started.
&prompt.root; <userinput>/etc/rc.d/dummy start Hello world!</userinput>
Greeting message: Hello world!</screen>
      </callout>

      <callout arearefs="rcng-args-kiss">
	<para>The same applies to any method our script provides,
	  not only to a standard one.  We have added a custom method
	  named <option>kiss</option>, and it can take advantage of
	  the extra arguments not less than <option>start</option>
	  does.  E.g.:</para>

	<screen>&prompt.root; <userinput>/etc/rc.d/dummy kiss</userinput>
A ghost gives you a kiss.
&prompt.root; <userinput>/etc/rc.d/dummy kiss Once I was Etaoin Shrdlu...</userinput>
A ghost gives you a kiss and whispers: Once I was Etaoin Shrdlu...</screen>
      </callout>

      <callout arearefs="rcng-args-all">
	<para>If we want just to pass all extra arguments to
	  any method, we can merely substitute <literal>"$@"</literal>
	  for <literal>"$1"</literal> in the last line of our script,
	  where we invoke <function>run_rc_command</function>.</para>

	<important>
	  <para>An &man.sh.1; programmer ought to understand the
	    subtle difference between <envar>$*</envar> and
	    <envar>$@</envar> as the ways to designate all positional
	    parameters.  For its in-depth discussion, refer to a
	    good handbook on &man.sh.1; scripting.  <emphasis>Do
	    not</emphasis> use the expressions until you fully
	    understand them because their misuse will result in
	    buggy and insecure scripts.</para>
	</important>

	<note>
	  <para>Currently <function>run_rc_command</function> may
	    have a bug that prevents it from keeping the original
	    boundaries between arguments.  That is, arguments with
	    embedded whitespace may not be processed correctly.
	    The bug stems from <envar>$*</envar> misuse.</para>
	</note>
      </callout>
    </calloutlist>
  </sect1>

  <sect1 id="rcng-furthur">
    <title>Further reading</title>

    <para><anchor id="lukem"/><ulink
      url="http://www.mewburn.net/luke/papers/rc.d.pdf">The original
      article by Luke Mewburn</ulink> offers a general overview of
      <filename>rc.d</filename> and detailed rationale for its
      design decisions.  It provides insight on the whole
      <filename>rc.d</filename> framework and its place in a modern
      BSD operating system.</para>

    <para><anchor id="manpages"/>The manual pages &man.rc.8;,
      &man.rc.subr.8;, and &man.rcorder.8; document the
      <filename>rc.d</filename> components in great detail.  You
      cannot fully use the <filename>rc.d</filename> power without
      studying the manual pages and referring to them while writing
      your own scripts.</para>

    <para>The major source of working, real-life examples is
      <filename>/etc/rc.d</filename> in a live system.  Its contents
      are easy and pleasant to read because most rough corners are
      hidden deep in &man.rc.subr.8;.  Keep in mind though that the
      <filename>/etc/rc.d</filename> scripts were not written by
      angels, so they might suffer from bugs and suboptimal design
      decisions.  Now you can improve them!</para>
  </sect1>
</article>