1.\" Copyright (c) 2010-2013 Hudson River Trading LLC 2.\" Written by: John H. Baldwin <jhb@FreeBSD.org> 3.\" All rights reserved. 4.\" 5.\" Redistribution and use in source and binary forms, with or without 6.\" modification, are permitted provided that the following conditions 7.\" are met: 8.\" 1. Redistributions of source code must retain the above copyright 9.\" notice, this list of conditions and the following disclaimer. 10.\" 2. Redistributions in binary form must reproduce the above copyright 11.\" notice, this list of conditions and the following disclaimer in the 12.\" documentation and/or other materials provided with the distribution. 13.\" 14.\" THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND 15.\" ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE 16.\" IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE 17.\" ARE DISCLAIMED. IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE 18.\" FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL 19.\" DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS 20.\" OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) 21.\" HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT 22.\" LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY 23.\" OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF 24.\" SUCH DAMAGE. 25.\" 26.Dd July 15, 2024 27.Dt ETCUPDATE 8 28.Os 29.Sh NAME 30.Nm etcupdate 31.Nd "manage updates to system files not updated by installworld" 32.Sh SYNOPSIS 33.Nm 34.Op Fl npBFN 35.Op Fl d Ar workdir 36.Op Fl r | Fl s Ar source | Fl t Ar tarball 37.Op Fl A Ar patterns 38.Op Fl D Ar destdir 39.Op Fl I Ar patterns 40.Op Fl L Ar logfile 41.Op Fl M Ar options 42.Op Fl m Ar make 43.Nm 44.Cm build 45.Op Fl BN 46.Op Fl d Ar workdir 47.Op Fl s Ar source 48.Op Fl L Ar logfile 49.Op Fl M Ar options 50.Op Fl m Ar make 51.Ar tarball 52.Nm 53.Cm diff 54.Op Fl d Ar workdir 55.Op Fl D Ar destdir 56.Op Fl I Ar patterns 57.Op Fl L Ar logfile 58.Nm 59.Cm extract 60.Op Fl BN 61.Op Fl d Ar workdir 62.Op Fl s Ar source | Fl t Ar tarball 63.Op Fl D Ar destdir 64.Op Fl L Ar logfile 65.Op Fl M Ar options 66.Op Fl m Ar make 67.Nm 68.Cm resolve 69.Op Fl p 70.Op Fl d Ar workdir 71.Op Fl D Ar destdir 72.Op Fl L Ar logfile 73.Nm 74.Cm revert 75.Op Fl d Ar workdir 76.Op Fl D Ar destdir 77.Op Fl L Ar logfile 78.Ar 79.Nm 80.Cm status 81.Op Fl d Ar workdir 82.Op Fl D Ar destdir 83.Sh DESCRIPTION 84The 85.Nm 86utility is a tool for managing updates to files that are not updated as 87part of 88.Sq make installworld 89such as files in 90.Pa /etc . 91It manages updates by doing a three-way merge of changes made to these 92files against the local versions. 93It is also designed to minimize the amount of user intervention with 94the goal of simplifying upgrades for clusters of machines. 95.Pp 96To perform a three-way merge, 97.Nm 98keeps copies of the current and previous versions of files that it manages. 99These copies are stored in two trees known as the 100.Dq current 101and 102.Dq previous 103trees. 104During a merge, 105.Nm 106compares the 107.Dq current 108and 109.Dq previous 110copies of each file to determine which changes need to be merged into the 111local version of each file. 112If a file can be updated without generating a conflict, 113.Nm 114will update the file automatically. 115If the local changes to a file conflict with the changes made to a file in 116the source tree, 117then a merge conflict is generated. 118The conflict must be resolved after the merge has finished. 119The 120.Nm 121utility will not perform a new merge until all conflicts from an earlier 122merge are resolved. 123.Sh MODES 124The 125.Nm 126utility supports several modes of operation. 127The mode is specified via an optional command argument. 128If present, the command must be the first argument on the command line. 129If a command is not specified, the default mode is used. 130.Ss Default Mode 131The default mode merges changes from the source tree to the destination 132directory. 133First, 134it updates the 135.Dq current 136and 137.Dq previous 138trees. 139Next, 140it compares the two trees merging changes into the destination directory. 141Finally, 142it displays warnings for any conditions it could not handle automatically. 143.Pp 144If the 145.Fl r 146option is not specified, 147then the first step taken is to update the 148.Dq current 149and 150.Dq previous 151trees. 152If a 153.Dq current 154tree already exists, 155then that tree is saved as the 156.Dq previous 157tree. 158An older 159.Dq previous 160tree is removed if it exists. 161By default the new 162.Dq current 163tree is built from a source tree. 164However, 165if a tarball is specified via the 166.Fl t 167option, 168then the tree is extracted from that tarball instead. 169.Pp 170Next, 171.Nm 172compares the files in the 173.Dq current 174and 175.Dq previous 176trees. 177If a file was removed from the 178.Dq current 179tree, 180then it will be removed from the destination directory only if it 181does not have any local modifications. 182If a file was added to the 183.Dq current 184tree, 185then it will be copied to the destination directory only if it 186would not clobber an existing file. 187If a file is changed in the 188.Dq current 189tree, 190then 191.Nm 192will attempt to merge the changes into the version of the file in the 193destination directory. 194If the merge encounters conflicts, 195then a version of the file with conflict markers will be saved for 196future resolution. 197If the merge does not encounter conflicts, 198then the merged version of the file will be saved in the destination 199directory. 200If 201.Nm 202is not able to safely merge in changes to a file other than a merge conflict, 203it will generate a warning. 204.Pp 205For each file that is updated a line will be output with a leading character 206to indicate the action taken. 207The possible actions follow: 208.Pp 209.Bl -tag -width "A" -compact -offset indent 210.It A 211Added 212.It C 213Conflict 214.It D 215Deleted 216.It M 217Merged 218.It U 219Updated 220.El 221.Pp 222Finally, 223if any warnings were encountered they are displayed after the merge has 224completed. 225.Pp 226Note that for certain files 227.Nm 228will perform post-install actions any time that the file is updated. 229Specifically, 230.Xr pwd_mkdb 8 231is invoked if 232.Pa /etc/master.passwd 233is changed, 234.Xr cap_mkdb 1 235is invoked to update 236.Pa /etc/login.conf.db 237if 238.Pa /etc/login.conf 239is changed, 240.Xr newaliases 1 241is invoked if 242.Pa /etc/mail/aliases 243is changed, 244.Xr services_mkdb 8 245is invoked if 246.Pa /etc/services 247is changed, 248.Xr tzsetup 8 249is invoked if 250.Pa /etc/localtime 251is changed and if 252.Fa /var/db/zoneinfo 253exists, 254and 255.Pa /etc/rc.d/motd 256is invoked if 257.Pa /etc/motd 258is changed. 259One exception is that if 260.Pa /etc/mail/aliases 261is changed and the destination directory is not the default, 262then a warning will be issued instead. 263This is due to a limitation of the 264.Xr newaliases 1 265command. 266Similarly, 267if 268.Pa /etc/motd 269is changed and the destination directory is not the default, 270then 271.Pa /etc/rc.d/motd 272will not be executed due to a limitation of that script. 273In this case no warning is issued as the result of 274.Pa /etc/rc.d/motd 275is merely cosmetic and will be corrected on the next reboot. 276.Ss Build Mode 277The 278.Cm build 279mode is used to build a tarball that contains a snapshot of a 280.Dq current 281tree. 282This tarball can be used by the default and extract modes. 283Using a tarball can allow 284.Nm 285to perform a merge without requiring a source tree that matches the 286currently installed world. 287The 288.Fa tarball 289argument specifies the name of the file to create. 290The file will be a 291.Xr tar 5 292file compressed with 293.Xr bzip2 1 . 294.Ss Diff Mode 295The 296.Cm diff 297mode compares the versions of files in the destination directory to the 298.Dq current 299tree and generates a unified format diff of the changes. 300This can be used to determine which files have been locally modified and how. 301Note that 302.Nm 303does not manage files that are not maintained in the source tree such as 304.Pa /etc/fstab 305and 306.Pa /etc/rc.conf . 307.Ss Extract Mode 308The 309.Cm extract 310mode generates a new 311.Dq current 312tree. 313Unlike the default mode, 314it does not save any existing 315.Dq current 316tree and does not modify any existing 317.Dq previous 318tree. 319The new 320.Dq current 321tree can either be built from a source tree or extracted from a tarball. 322.Ss Resolve Mode 323The 324.Cm resolve 325mode is used to resolve any conflicts encountered during a merge. 326In this mode, 327.Nm 328iterates over any existing conflicts prompting the user for actions to take 329on each conflicted file. 330For each file, the following actions are available: 331.Pp 332.Bl -tag -width "(tf) theirs-full" -compact 333.It (p) postpone 334Ignore this conflict for now. 335.It (df) diff-full 336Show all changes made to the merged file as a unified diff. 337.It (e) edit 338Change the merged file in an editor. 339.It (r) resolved 340Install the merged version of the file into the destination directory. 341.It (mf) mine-full 342Use the version of the file in the destination directory and ignore any 343changes made to the file in the 344.Dq current 345tree. 346.It (tf) theirs-full 347Use the version of the file from the 348.Dq current 349tree and discard any local changes made to the file. 350.It (h) help 351Display the list of commands. 352.El 353.Ss Revert Mode 354The 355.Cm revert 356mode is used to restore the stock versions of files. 357In this mode, 358.Nm 359installs the stock version of requested files. 360This mode cannot be used to restore directories, only individual files. 361.Ss Status Mode 362The 363.Cm status 364mode shows a summary of the results of the most recent merge. 365First it lists any files for which there are unresolved conflicts. 366Next it lists any warnings generated during the last merge. 367If the last merge did not generate any conflicts or warnings, 368then nothing will be output. 369.Sh OPTIONS 370The following options are available. 371Note that most options do not apply to all modes. 372.Bl -tag -width ".Fl A Ar patterns" 373.It Fl A Ar patterns 374Always install the new version of any files that match any of the patterns 375listed in 376.Ar patterns . 377Each pattern is evaluated as an 378.Xr sh 1 379shell pattern. 380This option may be specified multiple times to specify multiple patterns. 381Multiple space-separated patterns may also be specified in a single 382option. 383Note that ignored files specified via the 384.Ev IGNORE_FILES 385variable or the 386.Fl I 387option will not be installed. 388.It Fl B 389Do not build generated files in a private object tree. 390Instead, 391reuse the generated files from a previously built object tree that matches 392the source tree. 393This can be useful to avoid gratuitous conflicts in 394.Xr sendmail 8 395configuration 396files when bootstrapping. 397It can also be useful for building a tarball that matches a specific 398world build. 399.It Fl D Ar destdir 400Specify an alternate destination directory as the target of a merge. 401This is analogous to the 402.Dv DESTDIR 403variable used with 404.Sq make installworld . 405The default destination directory is an empty string which results in 406merges updating 407.Pa /etc 408on the local machine. 409.It Fl d Ar workdir 410Specify an alternate directory to use as the work directory. 411The work directory is used to store the 412.Dq current 413and 414.Dq previous 415trees as well as unresolved conflicts. 416The default work directory is 417.Pa <destdir>/var/db/etcupdate . 418.It Fl F 419Ignore changes in the FreeBSD ID string when comparing files in the 420destination directory to files in either of the 421.Dq current 422or 423.Dq previous 424trees. 425In 426.Cm diff 427mode, 428this reduces noise due to FreeBSD ID string changes in the output. 429During an update this can simplify handling for harmless conflicts caused 430by FreeBSD ID string changes. 431.Pp 432Specifically, 433if a file in the destination directory is identical to the same file in the 434.Dq previous 435tree modulo the FreeBSD ID string, 436then the file is treated as if it was unmodified and the 437.Dq current 438version of the file will be installed. 439Similarly, 440if a file in the destination directory is identical to the same file in the 441.Dq current 442tree modulo the FreeBSD ID string, 443then the 444.Dq current 445version of the file will be installed to update the ID string. 446If the 447.Dq previous 448and 449.Dq current 450versions of the file are identical, 451then 452.Nm 453will not change the file in the destination directory. 454.Pp 455Due to limitations in the 456.Xr diff 1 457command, 458this option may not have an effect if there are other changes in a file that 459are close to the FreeBSD ID string. 460.It Fl I Ar patterns 461Ignore any files that match any of the patterns listed in 462.Ar patterns . 463No warnings or other messages will be generated for those files during a 464merge. 465Each pattern is evaluated as an 466.Xr sh 1 467shell pattern. 468This option may be specified multiple times to specify multiple patterns. 469Multiple space-separated patterns may also be specified in a single 470option. 471.It Fl L Ar logfile 472Specify an alternate path for the log file. 473The 474.Nm 475utility logs each command that it invokes along with the standard output 476and standard error to this file. 477By default the log file is stored in a file named 478.Pa log 479in the work directory. 480.It Fl M Ar options 481Pass 482.Ar options 483as additional parameters to 484.Xr make 1 485when building a 486.Dq current 487tree. 488This can be used to set the 489.Dv TARGET 490or 491.Dv TARGET_ARCH 492variables for a cross-build. 493.It Fl m Ar make 494Use 495.Ar make 496as the 497.Xr make 1 498binary when building a 499.Dq current 500tree. 501.It Fl n 502Enable 503.Dq dry-run 504mode. 505Do not merge any changes to the destination directory. 506Instead, 507report what actions would be taken during a merge. 508Note that the existing 509.Dq current 510and 511.Dq previous 512trees will not be changed. 513If the 514.Fl r 515option is not specified, 516then a temporary 517.Dq current 518tree will be extracted to perform the comparison. 519.It Fl N 520Perform a 521.Dv NO_ROOT 522build when building a 523.Dq current 524tree. 525The resulting tree will include a corresponding 526.Pa METALOG 527file at its root. 528.It Fl p 529Enable 530.Dq pre-world 531mode. 532Only merge changes to files that are necessary to successfully run 533.Sq make installworld 534or 535.Sq make installkernel . 536When this flag is enabled, 537the existing 538.Dq current 539and 540.Dq previous 541trees are left alone. 542Instead, 543a temporary tree is populated with the necessary files. 544This temporary tree is compared against the 545.Dq current 546tree. 547This allows a normal update to be run after 548.Sq make installworld 549has completed. 550Any conflicts generated during a 551.Dq pre-world 552update should be resolved by a 553.Dq pre-world 554.Cm resolve . 555.It Fl r 556Do not update the 557.Dq current 558and 559.Dq previous 560trees during a merge. 561This can be used to 562.Dq re-run 563a previous merge operation. 564.It Fl s Ar source 565Specify an alternate source tree to use when building or extracting a 566.Dq current 567tree. 568The default source tree is 569.Pa /usr/src . 570.It Fl t Ar tarball 571Extract a new 572.Dq current 573tree from a tarball previously generated by the 574.Cm build 575command rather than building the tree from a source tree. 576.El 577.Sh CONFIG FILE 578The 579.Nm 580utility can also be configured by setting variables in an optional 581configuration file named 582.Pa /etc/etcupdate.conf . 583Note that command line options override settings in the configuration file. 584The configuration file is executed by 585.Xr sh 1 , 586so it uses that syntax to set configuration variables. 587The following variables can be set: 588.Bl -tag -width ".Ev ALWAYS_INSTALL" 589.It Ev ALWAYS_INSTALL 590Always install files that match any of the patterns listed in this variable 591similar to the 592.Fl A 593option. 594.It Ev DESTDIR 595Specify an alternate destination directory similar to the 596.Fl D 597option. 598.It Ev EDITOR 599Specify a program to edit merge conflicts. 600.It Ev FREEBSD_ID 601Ignore changes in the FreeBSD ID string similar to the 602.Fl F 603option. 604This is enabled by setting the variable to a non-empty value. 605.It Ev IGNORE_FILES 606Ignore files that match any of the patterns listed in this variable 607similar to the 608.Fl I 609option. 610.It Ev LOGFILE 611Specify an alternate path for the log file similar to the 612.Fl L 613option. 614.It Ev MAKE_CMD 615Specify the 616.Xr make 1 617binary when building a 618.Dq current 619tree similar to the 620.Fl m 621option. 622.It Ev MAKE_OPTIONS 623Pass additional options to 624.Xr make 1 625when building a 626.Dq current 627tree similar to the 628.Fl M 629option. 630.It Ev SRCDIR 631Specify an alternate source tree similar to the 632.Fl s 633option. 634.It Ev WORKDIR 635Specify an alternate work directory similar to the 636.Fl d 637option. 638.El 639.Sh ENVIRONMENT 640The 641.Nm 642utility uses the program identified in the 643.Ev EDITOR 644environment variable to edit merge conflicts. 645If 646.Ev EDITOR 647is not set, 648.Xr vi 1 649is used as the default editor. 650.Sh FILES 651.Bl -tag -width ".Pa /var/db/etcupdate/log" -compact 652.It Pa /etc/etcupdate.conf 653Optional config file. 654.It Pa /var/db/etcupdate 655Default work directory used to store trees and other data. 656.It Pa /var/db/etcupdate/log 657Default log file. 658.El 659.Sh EXIT STATUS 660.Ex -std 661.Sh EXAMPLES 662To compare the files in 663.Pa /etc 664with the stock versions: 665.Pp 666.Dl "etcupdate diff" 667.Pp 668To merge changes after an upgrade via the buildworld and installworld process: 669.Pp 670.Dl "etcupdate" 671.Pp 672To resolve any conflicts generated during a merge: 673.Pp 674.Dl "etcupdate resolve" 675.Ss Bootstrapping 676The 677.Nm 678utility may need to be bootstrapped before it can be used. 679The 680.Cm diff 681command will fail with an error about a missing reference tree if 682bootstrapping is needed. 683.Pp 684Bootstrapping 685.Nm 686requires a source tree that matches the currently installed world. 687The easiest way to ensure this is to bootstrap 688.Nm 689before updating the source tree to start the next world upgrade cycle. 690First, 691generate a reference tree: 692.Pp 693.Dl "etcupdate extract" 694.Pp 695Second, 696use the 697.Cm diff 698command to compare the reference tree to your current files in 699.Pa /etc . 700Undesired differences should be removed using an editor, 701.Xr patch 1 , 702or by copying files from the reference tree 703.Po 704located at 705.Pa /var/db/etcupdate/current 706by default 707.Pc 708. 709.Pp 710If the tree at 711.Pa /usr/src 712is already newer than the currently installed world, 713a new tree matching the currently installed world can be checked out to 714a temporary location. 715The reference tree for 716.Nm 717can then be generated via: 718.Pp 719.Dl "etcupdate extract -s /path/to/tree" 720.Pp 721The 722.Cm diff 723command can be used as above to remove undesired differences. 724Afterwards, 725the changes in the tree at 726.Pa /usr/src 727can be merged via a regular merge. 728.Sh DIAGNOSTICS 729The following warning messages may be generated during a merge. 730Note that several of these warnings cover obscure cases that should occur 731rarely if at all in practice. 732For example, 733if a file changes from a file to a directory in the 734.Dq current 735tree 736and the file was modified in the destination directory, 737then a warning will be triggered. 738In general, 739when a warning references a pathname, 740the corresponding file in the destination directory is not changed by a 741merge operation. 742.Bl -diag 743.It "Directory mismatch: <path> (<type>)" 744An attempt was made to create a directory at 745.Pa path 746but an existing file of type 747.Dq type 748already exists for that path name. 749.It "Modified link changed: <file> (<old> became <new>)" 750The target of a symbolic link named 751.Pa file 752was changed from 753.Dq old 754to 755.Dq new 756in the 757.Dq current 758tree. 759The symbolic link has been modified to point to a target that is neither 760.Dq old 761nor 762.Dq new 763in the destination directory. 764.It "Modified mismatch: <file> (<new> vs <dest>)" 765A file named 766.Pa file 767of type 768.Dq new 769was modified in the 770.Dq current 771tree, 772but the file exists as a different type 773.Dq dest 774in the destination directory. 775.It "Modified <type> changed: <file> (<old> became <new>)" 776A file named 777.Pa file 778changed type from 779.Dq old 780in the 781.Dq previous 782tree to type 783.Dq new 784in the 785.Dq current 786tree. 787The file in the destination directory of type 788.Dq type 789has been modified, 790so it could not be merged automatically. 791.It "Modified <type> remains: <file>" 792The file of type 793.Dq type 794named 795.Pa file 796has been removed from the 797.Dq current 798tree, 799but it has been locally modified. 800The modified version of the file remains in the destination directory. 801.It "Needs update: /etc/localtime (required manual update via tzsetup(8))" 802The 803.Fa /var/db/zoneinfo 804file does not exist, 805so 806.Nm 807was not able to refresh 808.Fa /etc/localtime 809from its source file in 810.Fa /usr/share/zoneinfo . 811Running 812.Xr tzsetup 8 813will both refresh 814.Fa /etc/localtime 815and generate 816.Fa /var/db/zoneinfo 817permitting future updates to refresh 818.Fa /etc/localtime 819automatically. 820.It "Needs update: /etc/mail/aliases.db (required manual update via newaliases(1))" 821The file 822.Pa /etc/mail/aliases 823was updated during a merge with a non-empty destination directory. 824Due to a limitation of the 825.Xr newaliases 1 826command, 827.Nm 828was not able to automatically update the corresponding aliases database. 829.It "New file mismatch: <file> (<new> vs <dest>)" 830A new file named 831.Pa file 832of type 833.Dq new 834has been added to the 835.Dq current 836tree. 837A file of that name already exists in the destination directory, 838but it is of a different type 839.Dq dest . 840.It "New link conflict: <file> (<new> vs <dest>)" 841A symbolic link named 842.Pa file 843has been added to the 844.Dq current 845tree that links to 846.Dq new . 847A symbolic link of the same name already exists in the destination 848directory, 849but it links to a different target 850.Dq dest . 851.It "Non-empty directory remains: <file>" 852The directory 853.Pa file 854was removed from the 855.Dq current 856tree, 857but it contains additional files in the destination directory. 858These additional files as well as the directory remain. 859.It "Remove mismatch: <file> (<old> became <new>)" 860A file named 861.Pa file 862changed from type 863.Dq old 864in the 865.Dq previous 866tree to type 867.Dq new 868in the 869.Dq current 870tree, 871but it has been removed in the destination directory. 872.It "Removed file changed: <file>" 873A file named 874.Pa file 875was modified in the 876.Dq current 877tree, 878but it has been removed in the destination directory. 879.It "Removed link changed: <file> (<old> became <new>)" 880The target of a symbolic link named 881.Pa file 882was changed from 883.Dq old 884to 885.Dq new 886in the 887.Dq current 888tree, 889but it has been removed in the destination directory. 890.El 891.Sh SEE ALSO 892.Xr cap_mkdb 1 , 893.Xr diff 1 , 894.Xr make 1 , 895.Xr newaliases 1 , 896.Xr sh 1 , 897.Xr pwd_mkdb 8 , 898.Xr services_mkdb 8 , 899.Xr tzsetup 8 900.Sh HISTORY 901The 902.Nm 903utility first appeared in 904.Fx 10.0 . 905.Sh AUTHORS 906The 907.Nm 908utility was written by 909.An John Baldwin Aq Mt jhb@FreeBSD.org . 910.Sh BUGS 911Rerunning a merge does not automatically delete conflicts left over from a 912previous merge. 913Any conflicts must be resolved before the merge can be rerun. 914It is not clear if this is a feature or a bug. 915.Pp 916There is no way to easily automate conflict resolution for specific files. 917For example, one can imagine a syntax along the lines of 918.Pp 919.Dl "etcupdate resolve tf /some/file" 920.Pp 921to resolve a specific conflict in an automated fashion. 922.Pp 923Bootstrapping 924.Nm 925often results in gratuitous diffs in 926.Pa /etc/mail/*.cf 927that cause conflicts in the first merge. 928If an object tree that matches the source tree is present when bootstrapping, 929then passing the 930.Fl B 931flag to the 932.Cm extract 933command can work around this. 934