xref: /illumos-gate/usr/src/cmd/sgs/libld/common/args.c (revision a99cb9618990662acbd3bab1b4a5b05a6ca62556)
1 /*
2  * CDDL HEADER START
3  *
4  * The contents of this file are subject to the terms of the
5  * Common Development and Distribution License (the "License").
6  * You may not use this file except in compliance with the License.
7  *
8  * You can obtain a copy of the license at usr/src/OPENSOLARIS.LICENSE
9  * or http://www.opensolaris.org/os/licensing.
10  * See the License for the specific language governing permissions
11  * and limitations under the License.
12  *
13  * When distributing Covered Code, include this CDDL HEADER in each
14  * file and include the License file at usr/src/OPENSOLARIS.LICENSE.
15  * If applicable, add the following below this CDDL HEADER, with the
16  * fields enclosed by brackets "[]" replaced with your own identifying
17  * information: Portions Copyright [yyyy] [name of copyright owner]
18  *
19  * CDDL HEADER END
20  */
21 
22 /*
23  *	Copyright (c) 1988 AT&T
24  *	  All Rights Reserved
25  *
26  * Copyright (c) 1989, 2010, Oracle and/or its affiliates. All rights reserved.
27  */
28 /*
29  * Copyright (c) 2012, Joyent, Inc.  All rights reserved.
30  * Copyright 2017 RackTop Systems.
31  */
32 
33 /*
34  * Publicly available flags are defined in ld(1).   The following flags are
35  * private, and may be removed at any time.
36  *
37  *    OPTION			MEANING
38  *
39  *    -z dtrace=symbol		assigns symbol to PT_SUNWDTRACE segment,
40  *				providing scratch area for dtrace processing.
41  *
42  *    -z noreloc		suppress relocation processing.  This provides
43  *				a mechanism for validating kernel module symbol
44  *				resolution that would normally incur fatal
45  *				relocation errors.
46  *
47  *    -z rtldinfo=symbol	assigns symbol to SUNW_RTLDINF dynamic tag,
48  *				providing pre-initialization specific routines
49  *				for TLS initialization.
50  *
51  *    -z nointerp		suppress the addition of an interpreter
52  *				section.  This is used to generate the kernel,
53  *				but makes no sense to be used by anyone else.
54  *
55  *    -z norelaxreloc		suppress the automatic addition of relaxed
56  *				relocations to GNU linkonce/COMDAT sections.
57  *
58  *    -z nosighandler		suppress the registration of the signal handler
59  *				used to manage SIGBUS.
60  */
61 
62 /*
63  * The following flags are committed, and will not be removed, but are
64  * not publically documented, either because they are obsolete, or because
65  * they exist to work around defects in other software and are not of
66  * sufficient interest otherwise.
67  *
68  *    OPTION			MEANING
69  *
70  *    -Wl,...			compiler drivers and configuration tools
71  *				have been known to pass this compiler option
72  *				to ld(1).  Strip off the "-Wl," prefix and
73  *			        process the remainder (...) as a normal option.
74  */
75 
76 #include	<sys/link.h>
77 #include	<stdio.h>
78 #include	<fcntl.h>
79 #include	<string.h>
80 #include	<errno.h>
81 #include	<elf.h>
82 #include	<unistd.h>
83 #include	<debug.h>
84 #include	"msg.h"
85 #include	"_libld.h"
86 
87 /*
88  * Define a set of local argument flags, the settings of these will be
89  * verified in check_flags() and lead to the appropriate output file flags
90  * being initialized.
91  */
92 typedef	enum {
93 	SET_UNKNOWN = -1,
94 	SET_FALSE = 0,
95 	SET_TRUE = 1
96 } Setstate;
97 
98 static Setstate	dflag	= SET_UNKNOWN;
99 static Setstate	zdflag	= SET_UNKNOWN;
100 static Setstate	Qflag	= SET_UNKNOWN;
101 static Setstate	Bdflag	= SET_UNKNOWN;
102 static Setstate zfwflag	= SET_UNKNOWN;
103 
104 static Boolean	aflag	= FALSE;
105 static Boolean	bflag	= FALSE;
106 static Boolean	sflag	= FALSE;
107 static Boolean	zinflag = FALSE;
108 static Boolean	zlflag	= FALSE;
109 static Boolean	Bgflag	= FALSE;
110 static Boolean	Blflag	= FALSE;
111 static Boolean	Beflag	= FALSE;
112 static Boolean	Bsflag	= FALSE;
113 static Boolean	Dflag	= FALSE;
114 static Boolean	Vflag	= FALSE;
115 
116 enum output_type {
117 	OT_RELOC,		/* relocatable object */
118 	OT_SHARED,		/* shared object */
119 	OT_EXEC,		/* dynamic executable */
120 	OT_KMOD,		/* kernel module */
121 };
122 
123 static enum output_type	otype = OT_EXEC;
124 
125 /*
126  * ztflag's state is set by pointing it to the matching string:
127  *	text | textoff | textwarn
128  */
129 static const char	*ztflag = NULL;
130 
131 /*
132  * Remember the guidance flags that result from the initial -z guidance
133  * option, so that they can be compared to any that follow. We only want
134  * to issue a warning when they differ.
135  */
136 static ofl_guideflag_t	initial_guidance_flags	= 0;
137 
138 static uintptr_t process_files_com(Ofl_desc *, int, char **);
139 static uintptr_t process_flags_com(Ofl_desc *, int, char **, int *);
140 
141 /*
142  * Print usage message to stderr - 2 modes, summary message only,
143  * and full usage message.
144  */
145 static void
146 usage_mesg(Boolean detail)
147 {
148 	(void) fprintf(stderr, MSG_INTL(MSG_ARG_USAGE),
149 	    MSG_ORIG(MSG_STR_OPTIONS));
150 
151 	if (detail == FALSE)
152 		return;
153 
154 	(void) fprintf(stderr, MSG_INTL(MSG_ARG_DETAIL_3));
155 	(void) fprintf(stderr, MSG_INTL(MSG_ARG_DETAIL_6));
156 	(void) fprintf(stderr, MSG_INTL(MSG_ARG_DETAIL_A));
157 	(void) fprintf(stderr, MSG_INTL(MSG_ARG_DETAIL_B));
158 	(void) fprintf(stderr, MSG_INTL(MSG_ARG_DETAIL_CBDR));
159 	(void) fprintf(stderr, MSG_INTL(MSG_ARG_DETAIL_CBDY));
160 	(void) fprintf(stderr, MSG_INTL(MSG_ARG_DETAIL_CBE));
161 	(void) fprintf(stderr, MSG_INTL(MSG_ARG_DETAIL_CBG));
162 	(void) fprintf(stderr, MSG_INTL(MSG_ARG_DETAIL_CBL));
163 	(void) fprintf(stderr, MSG_INTL(MSG_ARG_DETAIL_CBR));
164 	(void) fprintf(stderr, MSG_INTL(MSG_ARG_DETAIL_CBS));
165 	(void) fprintf(stderr, MSG_INTL(MSG_ARG_DETAIL_C));
166 	(void) fprintf(stderr, MSG_INTL(MSG_ARG_DETAIL_CC));
167 	(void) fprintf(stderr, MSG_INTL(MSG_ARG_DETAIL_D));
168 	(void) fprintf(stderr, MSG_INTL(MSG_ARG_DETAIL_CD));
169 	(void) fprintf(stderr, MSG_INTL(MSG_ARG_DETAIL_E));
170 	(void) fprintf(stderr, MSG_INTL(MSG_ARG_DETAIL_F));
171 	(void) fprintf(stderr, MSG_INTL(MSG_ARG_DETAIL_CF));
172 	(void) fprintf(stderr, MSG_INTL(MSG_ARG_DETAIL_CG));
173 	(void) fprintf(stderr, MSG_INTL(MSG_ARG_DETAIL_H));
174 	(void) fprintf(stderr, MSG_INTL(MSG_ARG_DETAIL_I));
175 	(void) fprintf(stderr, MSG_INTL(MSG_ARG_DETAIL_CI));
176 	(void) fprintf(stderr, MSG_INTL(MSG_ARG_DETAIL_L));
177 	(void) fprintf(stderr, MSG_INTL(MSG_ARG_DETAIL_CL));
178 	(void) fprintf(stderr, MSG_INTL(MSG_ARG_DETAIL_M));
179 	(void) fprintf(stderr, MSG_INTL(MSG_ARG_DETAIL_CM));
180 	(void) fprintf(stderr, MSG_INTL(MSG_ARG_DETAIL_CN));
181 	(void) fprintf(stderr, MSG_INTL(MSG_ARG_DETAIL_O));
182 	(void) fprintf(stderr, MSG_INTL(MSG_ARG_DETAIL_P));
183 	(void) fprintf(stderr, MSG_INTL(MSG_ARG_DETAIL_CP));
184 	(void) fprintf(stderr, MSG_INTL(MSG_ARG_DETAIL_CQ));
185 	(void) fprintf(stderr, MSG_INTL(MSG_ARG_DETAIL_R));
186 	(void) fprintf(stderr, MSG_INTL(MSG_ARG_DETAIL_CR));
187 	(void) fprintf(stderr, MSG_INTL(MSG_ARG_DETAIL_S));
188 	(void) fprintf(stderr, MSG_INTL(MSG_ARG_DETAIL_CS));
189 	(void) fprintf(stderr, MSG_INTL(MSG_ARG_DETAIL_T));
190 	(void) fprintf(stderr, MSG_INTL(MSG_ARG_DETAIL_U));
191 	(void) fprintf(stderr, MSG_INTL(MSG_ARG_DETAIL_CV));
192 	(void) fprintf(stderr, MSG_INTL(MSG_ARG_DETAIL_CY));
193 	(void) fprintf(stderr, MSG_INTL(MSG_ARG_DETAIL_ZA));
194 	(void) fprintf(stderr, MSG_INTL(MSG_ARG_DETAIL_ZAE));
195 	(void) fprintf(stderr, MSG_INTL(MSG_ARG_DETAIL_ZAL));
196 	(void) fprintf(stderr, MSG_INTL(MSG_ARG_DETAIL_ZADLIB));
197 	(void) fprintf(stderr, MSG_INTL(MSG_ARG_DETAIL_ZC));
198 	(void) fprintf(stderr, MSG_INTL(MSG_ARG_DETAIL_ZDEF));
199 	(void) fprintf(stderr, MSG_INTL(MSG_ARG_DETAIL_ZDFS));
200 	(void) fprintf(stderr, MSG_INTL(MSG_ARG_DETAIL_ZDRS));
201 	(void) fprintf(stderr, MSG_INTL(MSG_ARG_DETAIL_ZE));
202 	(void) fprintf(stderr, MSG_INTL(MSG_ARG_DETAIL_ZFATW));
203 	(void) fprintf(stderr, MSG_INTL(MSG_ARG_DETAIL_ZFA));
204 	(void) fprintf(stderr, MSG_INTL(MSG_ARG_DETAIL_ZGP));
205 	(void) fprintf(stderr, MSG_INTL(MSG_ARG_DETAIL_ZGUIDE));
206 	(void) fprintf(stderr, MSG_INTL(MSG_ARG_DETAIL_ZH));
207 	(void) fprintf(stderr, MSG_INTL(MSG_ARG_DETAIL_ZIG));
208 	(void) fprintf(stderr, MSG_INTL(MSG_ARG_DETAIL_ZINA));
209 	(void) fprintf(stderr, MSG_INTL(MSG_ARG_DETAIL_ZINI));
210 	(void) fprintf(stderr, MSG_INTL(MSG_ARG_DETAIL_ZINT));
211 	(void) fprintf(stderr, MSG_INTL(MSG_ARG_DETAIL_ZLAZY));
212 	(void) fprintf(stderr, MSG_INTL(MSG_ARG_DETAIL_ZLD32));
213 	(void) fprintf(stderr, MSG_INTL(MSG_ARG_DETAIL_ZLD64));
214 	(void) fprintf(stderr, MSG_INTL(MSG_ARG_DETAIL_ZLO));
215 	(void) fprintf(stderr, MSG_INTL(MSG_ARG_DETAIL_ZM));
216 	(void) fprintf(stderr, MSG_INTL(MSG_ARG_DETAIL_ZNC));
217 	(void) fprintf(stderr, MSG_INTL(MSG_ARG_DETAIL_ZNDFS));
218 	(void) fprintf(stderr, MSG_INTL(MSG_ARG_DETAIL_ZNDEF));
219 	(void) fprintf(stderr, MSG_INTL(MSG_ARG_DETAIL_ZNDEL));
220 	(void) fprintf(stderr, MSG_INTL(MSG_ARG_DETAIL_ZNDLO));
221 	(void) fprintf(stderr, MSG_INTL(MSG_ARG_DETAIL_ZNDU));
222 	(void) fprintf(stderr, MSG_INTL(MSG_ARG_DETAIL_ZNLD));
223 	(void) fprintf(stderr, MSG_INTL(MSG_ARG_DETAIL_ZNOW));
224 	(void) fprintf(stderr, MSG_INTL(MSG_ARG_DETAIL_ZNPA));
225 	(void) fprintf(stderr, MSG_INTL(MSG_ARG_DETAIL_ZNV));
226 	(void) fprintf(stderr, MSG_INTL(MSG_ARG_DETAIL_ZO));
227 	(void) fprintf(stderr, MSG_INTL(MSG_ARG_DETAIL_ZPIA));
228 	(void) fprintf(stderr, MSG_INTL(MSG_ARG_DETAIL_ZRL));
229 	(void) fprintf(stderr, MSG_INTL(MSG_ARG_DETAIL_ZRREL));
230 	(void) fprintf(stderr, MSG_INTL(MSG_ARG_DETAIL_ZRS));
231 	(void) fprintf(stderr, MSG_INTL(MSG_ARG_DETAIL_ZRSN));
232 	(void) fprintf(stderr, MSG_INTL(MSG_ARG_DETAIL_ZRSGRP));
233 	(void) fprintf(stderr, MSG_INTL(MSG_ARG_DETAIL_ZSCAP));
234 	(void) fprintf(stderr, MSG_INTL(MSG_ARG_DETAIL_ZTARG));
235 	(void) fprintf(stderr, MSG_INTL(MSG_ARG_DETAIL_ZT));
236 	(void) fprintf(stderr, MSG_INTL(MSG_ARG_DETAIL_ZTO));
237 	(void) fprintf(stderr, MSG_INTL(MSG_ARG_DETAIL_ZTW));
238 	(void) fprintf(stderr, MSG_INTL(MSG_ARG_DETAIL_ZTY));
239 	(void) fprintf(stderr, MSG_INTL(MSG_ARG_DETAIL_ZWRAP));
240 	(void) fprintf(stderr, MSG_INTL(MSG_ARG_DETAIL_ZVER));
241 }
242 
243 /*
244  * Rescan the archives seen on the command line in order
245  * to handle circularly dependent archives, stopping when
246  * no further member extraction occurs.
247  *
248  * entry:
249  *	ofl - Output file descriptor
250  *	isgrp - True if this is a an archive group search, False
251  *		to search starting with argv[1] through end_arg_ndx
252  *	end_arg_ndx - Index of final argv element to consider.
253  */
254 static uintptr_t
255 ld_rescan_archives(Ofl_desc *ofl, int isgrp, int end_arg_ndx)
256 {
257 	ofl->ofl_flags1 |= FLG_OF1_EXTRACT;
258 
259 	while (ofl->ofl_flags1 & FLG_OF1_EXTRACT) {
260 		Aliste		idx;
261 		Ar_desc		*adp;
262 		Word		start_ndx = isgrp ? ofl->ofl_ars_gsndx : 0;
263 		Word		ndx = 0;
264 
265 		ofl->ofl_flags1 &= ~FLG_OF1_EXTRACT;
266 
267 		DBG_CALL(Dbg_file_ar_rescan(ofl->ofl_lml,
268 		    isgrp ? ofl->ofl_ars_gsandx : 1, end_arg_ndx));
269 
270 		for (APLIST_TRAVERSE(ofl->ofl_ars, idx, adp)) {
271 			/* If not to starting index yet, skip it */
272 			if (ndx++ < start_ndx)
273 				continue;
274 
275 			/*
276 			 * If this archive was processed with -z allextract,
277 			 * then all members have already been extracted.
278 			 */
279 			if (adp->ad_elf == NULL)
280 				continue;
281 
282 			/*
283 			 * Reestablish any archive specific command line flags.
284 			 */
285 			ofl->ofl_flags1 &= ~MSK_OF1_ARCHIVE;
286 			ofl->ofl_flags1 |= (adp->ad_flags & MSK_OF1_ARCHIVE);
287 
288 			/*
289 			 * Re-process the archive.  Note that a file descriptor
290 			 * is unnecessary, as the file is already available in
291 			 * memory.
292 			 */
293 			if (!ld_process_archive(adp->ad_name, -1, adp, ofl))
294 				return (S_ERROR);
295 			if (ofl->ofl_flags & FLG_OF_FATAL)
296 				return (1);
297 		}
298 	}
299 
300 	return (1);
301 }
302 
303 /*
304  * Checks the command line option flags for consistency.
305  */
306 static uintptr_t
307 check_flags(Ofl_desc * ofl, int argc)
308 {
309 	/*
310 	 * If the user specified -zguidance=noall, then we can safely disable
311 	 * the entire feature. The purpose of -zguidance=noall is to allow
312 	 * the user to override guidance specified from a makefile via
313 	 * the LD_OPTIONS environment variable, and so, we want to behave
314 	 * in exactly the same manner we would have if no option were present.
315 	 */
316 	if ((ofl->ofl_guideflags & (FLG_OFG_ENABLE | FLG_OFG_NO_ALL)) ==
317 	    (FLG_OFG_ENABLE | FLG_OFG_NO_ALL))
318 		ofl->ofl_guideflags &= ~FLG_OFG_ENABLE;
319 
320 	if (Plibpath && (Llibdir || Ulibdir))
321 		ld_eprintf(ofl, ERR_FATAL, MSG_INTL(MSG_ARG_YP),
322 		    Llibdir ? 'L' : 'U');
323 
324 	if ((otype == OT_RELOC) || (otype == OT_KMOD)) {
325 		if (otype == OT_RELOC) {
326 			if (dflag == SET_UNKNOWN)
327 				dflag = SET_FALSE;
328 			if ((dflag == SET_TRUE) &&
329 			    OFL_GUIDANCE(ofl, FLG_OFG_NO_KMOD)) {
330 				ld_eprintf(ofl, ERR_GUIDANCE,
331 				    MSG_INTL(MSG_GUIDE_KMOD));
332 			}
333 		} else if (otype == OT_KMOD) {
334 			if (dflag != SET_UNKNOWN) {
335 				ld_eprintf(ofl, ERR_FATAL,
336 				    MSG_INTL(MSG_MARG_INCOMP),
337 				    MSG_INTL(MSG_MARG_TYPE_KMOD),
338 				    MSG_ORIG(MSG_ARG_D));
339 			}
340 
341 			dflag = SET_TRUE;
342 		}
343 
344 		/*
345 		 * Combining relocations when building a relocatable
346 		 * object isn't allowed.  Warn the user, but proceed.
347 		 */
348 		if (ofl->ofl_flags & FLG_OF_COMREL) {
349 			const char *msg;
350 
351 			if (otype == OT_RELOC) {
352 				msg = MSG_INTL(MSG_MARG_REL);
353 			} else {
354 				msg = MSG_INTL(MSG_MARG_TYPE_KMOD);
355 			}
356 			ld_eprintf(ofl, ERR_WARNING, MSG_INTL(MSG_MARG_INCOMP),
357 			    msg,
358 			    MSG_ORIG(MSG_ARG_ZCOMBRELOC));
359 		}
360 		ofl->ofl_flags |= FLG_OF_RELOBJ;
361 
362 		if (otype == OT_KMOD)
363 			ofl->ofl_flags |= FLG_OF_KMOD;
364 	} else {
365 		/*
366 		 * Translating object capabilities to symbol capabilities is
367 		 * only meaningful when creating a relocatable object.
368 		 */
369 		if (ofl->ofl_flags & FLG_OF_OTOSCAP)
370 			ld_eprintf(ofl, ERR_FATAL, MSG_INTL(MSG_MARG_ONLY),
371 			    MSG_ORIG(MSG_ARG_ZSYMBOLCAP),
372 			    MSG_INTL(MSG_MARG_REL));
373 
374 		/*
375 		 * If the user hasn't explicitly requested that relocations
376 		 * not be combined, combine them by default.
377 		 */
378 		if ((ofl->ofl_flags & FLG_OF_NOCOMREL) == 0)
379 			ofl->ofl_flags |= FLG_OF_COMREL;
380 	}
381 
382 	if (zdflag == SET_TRUE)
383 		ofl->ofl_flags |= FLG_OF_NOUNDEF;
384 
385 	if (zinflag)
386 		ofl->ofl_dtflags_1 |= DF_1_INTERPOSE;
387 
388 	if (sflag)
389 		ofl->ofl_flags |= FLG_OF_STRIP;
390 
391 	if (Qflag == SET_TRUE)
392 		ofl->ofl_flags |= FLG_OF_ADDVERS;
393 
394 	if (Blflag)
395 		ofl->ofl_flags |= FLG_OF_AUTOLCL;
396 
397 	if (Beflag)
398 		ofl->ofl_flags |= FLG_OF_AUTOELM;
399 
400 	if (Blflag && Beflag)
401 		ld_eprintf(ofl, ERR_FATAL, MSG_INTL(MSG_ARG_INCOMP),
402 		    MSG_ORIG(MSG_ARG_BELIMINATE), MSG_ORIG(MSG_ARG_BLOCAL));
403 
404 	if (ofl->ofl_interp && (ofl->ofl_flags1 & FLG_OF1_NOINTRP))
405 		ld_eprintf(ofl, ERR_FATAL, MSG_INTL(MSG_ARG_INCOMP),
406 		    MSG_ORIG(MSG_ARG_CI), MSG_ORIG(MSG_ARG_ZNOINTERP));
407 
408 	if ((ofl->ofl_flags1 & (FLG_OF1_NRLXREL | FLG_OF1_RLXREL)) ==
409 	    (FLG_OF1_NRLXREL | FLG_OF1_RLXREL))
410 		ld_eprintf(ofl, ERR_FATAL, MSG_INTL(MSG_ARG_INCOMP),
411 		    MSG_ORIG(MSG_ARG_ZRELAXRELOC),
412 		    MSG_ORIG(MSG_ARG_ZNORELAXRELOC));
413 
414 	if (ofl->ofl_filtees && (otype != OT_SHARED))
415 		ld_eprintf(ofl, ERR_FATAL, MSG_INTL(MSG_MARG_ST_ONLYAVL),
416 		    ((ofl->ofl_flags & FLG_OF_AUX) ?
417 		    MSG_INTL(MSG_MARG_FILTER_AUX) : MSG_INTL(MSG_MARG_FILTER)));
418 
419 	if (dflag != SET_FALSE) {
420 		/*
421 		 * Set -Bdynamic on by default, setting is rechecked as input
422 		 * files are processed.
423 		 */
424 		ofl->ofl_flags |=
425 		    (FLG_OF_DYNAMIC | FLG_OF_DYNLIBS | FLG_OF_PROCRED);
426 
427 		if (aflag)
428 			ld_eprintf(ofl, ERR_FATAL, MSG_INTL(MSG_ARG_INCOMP),
429 			    MSG_ORIG(MSG_ARG_DY), MSG_ORIG(MSG_ARG_A));
430 
431 		if (bflag)
432 			ofl->ofl_flags |= FLG_OF_BFLAG;
433 
434 		if (Bgflag == TRUE) {
435 			if (zdflag == SET_FALSE)
436 				ld_eprintf(ofl, ERR_FATAL,
437 				    MSG_INTL(MSG_ARG_INCOMP),
438 				    MSG_ORIG(MSG_ARG_BGROUP),
439 				    MSG_ORIG(MSG_ARG_ZNODEF));
440 			ofl->ofl_dtflags_1 |= DF_1_GROUP;
441 			ofl->ofl_flags |= FLG_OF_NOUNDEF;
442 		}
443 
444 		/*
445 		 * If the use of default library searching has been suppressed
446 		 * but no runpaths have been provided we're going to have a hard
447 		 * job running this object.
448 		 */
449 		if ((ofl->ofl_dtflags_1 & DF_1_NODEFLIB) && !ofl->ofl_rpath)
450 			ld_eprintf(ofl, ERR_WARNING, MSG_INTL(MSG_ARG_NODEFLIB),
451 			    MSG_INTL(MSG_MARG_RPATH));
452 
453 		/*
454 		 * By default, text relocation warnings are given when building
455 		 * an executable unless the -b flag is specified.  This option
456 		 * implies that unclean text can be created, so no warnings are
457 		 * generated unless specifically asked for.
458 		 */
459 		if ((ztflag == MSG_ORIG(MSG_ARG_ZTEXTOFF)) ||
460 		    ((ztflag == NULL) && bflag)) {
461 			ofl->ofl_flags1 |= FLG_OF1_TEXTOFF;
462 			ofl->ofl_guideflags |= FLG_OFG_NO_TEXT;
463 		} else if (ztflag == MSG_ORIG(MSG_ARG_ZTEXT)) {
464 			ofl->ofl_flags |= FLG_OF_PURETXT;
465 			ofl->ofl_guideflags |= FLG_OFG_NO_TEXT;
466 		}
467 
468 		if ((otype == OT_SHARED) || (otype == OT_EXEC)) {
469 			/*
470 			 * Create a dynamic object.  -Bdirect indicates that all
471 			 * references should be bound directly.  This also
472 			 * enables lazyloading.  Individual symbols can be
473 			 * bound directly (or not) using mapfiles and the
474 			 * DIRECT (NODIRECT) qualifier.  With this capability,
475 			 * each syminfo entry is tagged SYMINFO_FLG_DIRECTBIND.
476 			 * Prior to this per-symbol direct binding, runtime
477 			 * direct binding was controlled via the DF_1_DIRECT
478 			 * flag.  This flag affected all references from the
479 			 * object.  -Bdirect continues to set this flag, and
480 			 * thus provides a means of taking a newly built
481 			 * direct binding object back to older systems.
482 			 *
483 			 * NOTE, any use of per-symbol NODIRECT bindings, or
484 			 * -znodirect, will disable the creation of the
485 			 * DF_1_DIRECT flag.  Older runtime linkers do not
486 			 * have the capability to do per-symbol direct bindings.
487 			 */
488 			if (Bdflag == SET_TRUE) {
489 				ofl->ofl_dtflags_1 |= DF_1_DIRECT;
490 				ofl->ofl_flags1 |= FLG_OF1_LAZYLD;
491 				ofl->ofl_guideflags |= FLG_OFG_NO_LAZY;
492 				ofl->ofl_flags |= FLG_OF_SYMINFO;
493 			}
494 
495 			/*
496 			 * -Bnodirect disables directly binding to any symbols
497 			 * exported from the object being created.  Individual
498 			 * references to external objects can still be affected
499 			 * by -zdirect or mapfile DIRECT directives.
500 			 */
501 			if (Bdflag == SET_FALSE) {
502 				ofl->ofl_flags1 |= (FLG_OF1_NDIRECT |
503 				    FLG_OF1_NGLBDIR | FLG_OF1_ALNODIR);
504 				ofl->ofl_flags |= FLG_OF_SYMINFO;
505 			}
506 		}
507 
508 		if (otype == OT_EXEC) {
509 			/*
510 			 * Dynamically linked executable.
511 			 */
512 			ofl->ofl_flags |= FLG_OF_EXEC;
513 
514 			if (zdflag != SET_FALSE)
515 				ofl->ofl_flags |= FLG_OF_NOUNDEF;
516 
517 			/*
518 			 * -z textwarn is the default for executables, and
519 			 * only an explicit -z text* option can change that,
520 			 * so there's no need to provide additional guidance.
521 			 */
522 			ofl->ofl_guideflags |= FLG_OFG_NO_TEXT;
523 
524 			if (Bsflag)
525 				ld_eprintf(ofl, ERR_FATAL,
526 				    MSG_INTL(MSG_ARG_DY_INCOMP),
527 				    MSG_ORIG(MSG_ARG_BSYMBOLIC));
528 			if (ofl->ofl_soname)
529 				ld_eprintf(ofl, ERR_FATAL,
530 				    MSG_INTL(MSG_MARG_DY_INCOMP),
531 				    MSG_INTL(MSG_MARG_SONAME));
532 		} else if (otype == OT_SHARED) {
533 			/*
534 			 * Shared library.
535 			 */
536 			ofl->ofl_flags |= FLG_OF_SHAROBJ;
537 
538 			/*
539 			 * By default, print text relocation warnings for
540 			 * executables but *not* for shared objects. However,
541 			 * if -z guidance is on, issue warnings for shared
542 			 * objects as well.
543 			 *
544 			 * If -z textwarn is explicitly specified, also issue
545 			 * guidance messages if -z guidance is on, but not
546 			 * for -z text or -z textoff.
547 			 */
548 			if (ztflag == NULL) {
549 				if (!OFL_GUIDANCE(ofl, FLG_OFG_NO_TEXT))
550 					ofl->ofl_flags1 |= FLG_OF1_TEXTOFF;
551 			} else if ((ofl->ofl_flags & FLG_OF_PURETXT) ||
552 			    (ofl->ofl_flags1 & FLG_OF1_TEXTOFF)) {
553 				ofl->ofl_guideflags |= FLG_OFG_NO_TEXT;
554 			}
555 
556 			if (Bsflag) {
557 				/*
558 				 * -Bsymbolic, and -Bnodirect make no sense.
559 				 */
560 				if (Bdflag == SET_FALSE)
561 					ld_eprintf(ofl, ERR_FATAL,
562 					    MSG_INTL(MSG_ARG_INCOMP),
563 					    MSG_ORIG(MSG_ARG_BSYMBOLIC),
564 					    MSG_ORIG(MSG_ARG_BNODIRECT));
565 				ofl->ofl_flags |= FLG_OF_SYMBOLIC;
566 				ofl->ofl_dtflags |= DF_SYMBOLIC;
567 			}
568 		} else {
569 			/*
570 			 * Dynamic relocatable object.
571 			 */
572 			if (ztflag == NULL)
573 				ofl->ofl_flags1 |= FLG_OF1_TEXTOFF;
574 			ofl->ofl_guideflags |= FLG_OFG_NO_TEXT;
575 
576 			if (ofl->ofl_interp)
577 				ld_eprintf(ofl, ERR_FATAL,
578 				    MSG_INTL(MSG_MARG_INCOMP),
579 				    MSG_INTL(MSG_MARG_REL),
580 				    MSG_ORIG(MSG_ARG_CI));
581 		}
582 
583 		assert((ofl->ofl_flags & (FLG_OF_SHAROBJ|FLG_OF_EXEC)) !=
584 		    (FLG_OF_SHAROBJ|FLG_OF_EXEC));
585 	} else {
586 		ofl->ofl_flags |= FLG_OF_STATIC;
587 
588 		if (bflag)
589 			ld_eprintf(ofl, ERR_FATAL, MSG_INTL(MSG_ARG_ST_INCOMP),
590 			    MSG_ORIG(MSG_ARG_B));
591 		if (ofl->ofl_soname)
592 			ld_eprintf(ofl, ERR_FATAL, MSG_INTL(MSG_MARG_ST_INCOMP),
593 			    MSG_INTL(MSG_MARG_SONAME));
594 		if (ofl->ofl_depaudit)
595 			ld_eprintf(ofl, ERR_FATAL, MSG_INTL(MSG_ARG_ST_INCOMP),
596 			    MSG_ORIG(MSG_ARG_CP));
597 		if (ofl->ofl_audit)
598 			ld_eprintf(ofl, ERR_FATAL, MSG_INTL(MSG_ARG_ST_INCOMP),
599 			    MSG_ORIG(MSG_ARG_P));
600 		if (ofl->ofl_config)
601 			ld_eprintf(ofl, ERR_FATAL, MSG_INTL(MSG_ARG_ST_INCOMP),
602 			    MSG_ORIG(MSG_ARG_C));
603 		if (ztflag)
604 			ld_eprintf(ofl, ERR_FATAL, MSG_INTL(MSG_ARG_ST_INCOMP),
605 			    MSG_ORIG(MSG_ARG_ZTEXTALL));
606 		if (otype == OT_SHARED)
607 			ld_eprintf(ofl, ERR_FATAL, MSG_INTL(MSG_MARG_ST_INCOMP),
608 			    MSG_INTL(MSG_MARG_SO));
609 		if (aflag && (otype == OT_RELOC))
610 			ld_eprintf(ofl, ERR_FATAL, MSG_INTL(MSG_MARG_INCOMP),
611 			    MSG_ORIG(MSG_ARG_A), MSG_INTL(MSG_MARG_REL));
612 
613 		if (otype == OT_RELOC) {
614 			/*
615 			 * We can only strip the symbol table and string table
616 			 * if no output relocations will refer to them.
617 			 */
618 			if (sflag)
619 				ld_eprintf(ofl, ERR_WARNING,
620 				    MSG_INTL(MSG_ARG_STRIP),
621 				    MSG_INTL(MSG_MARG_REL),
622 				    MSG_INTL(MSG_MARG_STRIP));
623 
624 			if (ztflag == NULL)
625 				ofl->ofl_flags1 |= FLG_OF1_TEXTOFF;
626 			ofl->ofl_guideflags |= FLG_OFG_NO_TEXT;
627 
628 			if (ofl->ofl_interp)
629 				ld_eprintf(ofl, ERR_FATAL,
630 				    MSG_INTL(MSG_MARG_INCOMP),
631 				    MSG_INTL(MSG_MARG_REL),
632 				    MSG_ORIG(MSG_ARG_CI));
633 		} else {
634 			/*
635 			 * Static executable.
636 			 */
637 			ofl->ofl_flags |= FLG_OF_EXEC | FLG_OF_PROCRED;
638 
639 			if (zdflag != SET_FALSE)
640 				ofl->ofl_flags |= FLG_OF_NOUNDEF;
641 		}
642 	}
643 
644 	/*
645 	 * If the user didn't supply an output file name supply a default.
646 	 */
647 	if (ofl->ofl_name == NULL)
648 		ofl->ofl_name = MSG_ORIG(MSG_STR_AOUT);
649 
650 	/*
651 	 * We set the entrance criteria after all input argument processing as
652 	 * it is only at this point we're sure what the output image will be
653 	 * (static or dynamic).
654 	 */
655 	if (ld_ent_setup(ofl, ld_targ.t_m.m_segm_align) == S_ERROR)
656 		return (S_ERROR);
657 
658 	/*
659 	 * Does the host currently running the linker have the same
660 	 * byte order as the target for which the object is being produced?
661 	 * If not, set FLG_OF1_ENCDIFF so relocation code will know
662 	 * to check.
663 	 */
664 	if (_elf_sys_encoding() != ld_targ.t_m.m_data)
665 		ofl->ofl_flags1 |= FLG_OF1_ENCDIFF;
666 
667 	/*
668 	 * If the target has special executable section filling requirements,
669 	 * register the fill function with libelf
670 	 */
671 	if (ld_targ.t_ff.ff_execfill != NULL)
672 		_elf_execfill(ld_targ.t_ff.ff_execfill);
673 
674 	/*
675 	 * Initialize string tables.  Symbol definitions within mapfiles can
676 	 * result in the creation of input sections.
677 	 */
678 	if (ld_init_strings(ofl) == S_ERROR)
679 		return (S_ERROR);
680 
681 	/*
682 	 * Process mapfiles. Mapfile can redefine or add sections/segments,
683 	 * so this must come after the default entrance criteria are established
684 	 * (above).
685 	 */
686 	if (ofl->ofl_maps) {
687 		const char	*name;
688 		Aliste		idx;
689 
690 		for (APLIST_TRAVERSE(ofl->ofl_maps, idx, name))
691 			if (!ld_map_parse(name, ofl))
692 				return (S_ERROR);
693 
694 		if (!ld_map_post_process(ofl))
695 			return (S_ERROR);
696 	}
697 
698 	/*
699 	 * If a mapfile has been used to define a single symbolic scope of
700 	 * interfaces, -Bsymbolic is established.  This global setting goes
701 	 * beyond individual symbol protection, and ensures all relocations
702 	 * (even those that reference section symbols) are processed within
703 	 * the object being built.
704 	 */
705 	if (((ofl->ofl_flags &
706 	    (FLG_OF_MAPSYMB | FLG_OF_MAPGLOB)) == FLG_OF_MAPSYMB) &&
707 	    (ofl->ofl_flags & (FLG_OF_AUTOLCL | FLG_OF_AUTOELM))) {
708 		ofl->ofl_flags |= FLG_OF_SYMBOLIC;
709 		ofl->ofl_dtflags |= DF_SYMBOLIC;
710 	}
711 
712 	/*
713 	 * If -zloadfltr is set, verify that filtering is in effect.  Filters
714 	 * are either established from the command line, and affect the whole
715 	 * object, or are set on a per-symbol basis from a mapfile.
716 	 */
717 	if (zlflag) {
718 		if ((ofl->ofl_filtees == NULL) && (ofl->ofl_dtsfltrs == NULL))
719 			ld_eprintf(ofl, ERR_FATAL, MSG_INTL(MSG_ARG_NOFLTR),
720 			    MSG_ORIG(MSG_ARG_ZLOADFLTR));
721 		ofl->ofl_dtflags_1 |= DF_1_LOADFLTR;
722 	}
723 
724 	/*
725 	 * Check that we have something to work with. This check is carried out
726 	 * after mapfile processing as its possible a mapfile is being used to
727 	 * define symbols, in which case it would be sufficient to build the
728 	 * output file purely from the mapfile.
729 	 */
730 	if ((ofl->ofl_objscnt == 0) && (ofl->ofl_soscnt == 0)) {
731 		if ((Vflag ||
732 		    (Dflag && (dbg_desc->d_extra & DBG_E_HELP_EXIT))) &&
733 		    (argc == 2)) {
734 			ofl->ofl_flags1 |= FLG_OF1_DONE;
735 		} else {
736 			ld_eprintf(ofl, ERR_FATAL, MSG_INTL(MSG_ARG_NOFILES));
737 			return (S_ERROR);
738 		}
739 	}
740 	return (1);
741 }
742 
743 /*
744  * Decompose the string pointed by optarg into argv[][] so that argv[][] can be
745  * used as an argument to getopt().
746  *
747  * If the second argument 'usage' is not NULL, then this is called from the
748  * first pass. Else this is called from the second pass.
749  */
750 static uintptr_t
751 createargv(Ofl_desc *ofl, int *usage)
752 {
753 	int		argc = 0, idx = 0, ooptind;
754 	uintptr_t	ret;
755 	char		**argv, *p0;
756 
757 	/*
758 	 * The argument being examined is either:
759 	 *	ld32=	or
760 	 *	ld64=
761 	 */
762 #if	defined(_LP64)
763 	if (optarg[2] == '3')
764 		return (0);
765 #else
766 	if (optarg[2] == '6')
767 		return (0);
768 #endif
769 
770 	p0 = &optarg[5];
771 
772 	/*
773 	 * Count the number of arguments.
774 	 */
775 	while (*p0) {
776 		/*
777 		 * Pointing at non-separator character.
778 		 */
779 		if (*p0 != ',') {
780 			argc++;
781 			while (*p0 && (*p0 != ','))
782 				p0++;
783 			continue;
784 		}
785 
786 		/*
787 		 * Pointing at a separator character.
788 		 */
789 		if (*p0 == ',') {
790 			while (*p0 == ',')
791 				p0++;
792 			continue;
793 		}
794 	}
795 
796 	if (argc == 0)
797 		return (0);
798 
799 	/*
800 	 * Allocate argument vector.
801 	 */
802 	if ((p0 = (char *)strdup(&optarg[5])) == NULL)
803 		return (S_ERROR);
804 	if ((argv = libld_malloc((sizeof (char *)) * (argc + 1))) == NULL)
805 		return (S_ERROR);
806 
807 	while (*p0) {
808 		char *p;
809 
810 		/*
811 		 * Pointing at the beginning of non-separator character string.
812 		 */
813 		if (*p0 != ',') {
814 			p = p0;
815 			while (*p0 && (*p0 != ','))
816 				p0++;
817 			argv[idx++] = p;
818 			if (*p0) {
819 				*p0 = '\0';
820 				p0++;
821 			}
822 			continue;
823 		}
824 
825 		/*
826 		 * Pointing at the beginining of separator character string.
827 		 */
828 		if (*p0 == ',') {
829 			while (*p0 == ',')
830 				p0++;
831 			continue;
832 		}
833 	}
834 	argv[idx] = 0;
835 	ooptind = optind;
836 	optind = 0;
837 
838 	/*
839 	 * Dispatch to pass1 or pass2
840 	 */
841 	if (usage)
842 		ret = process_flags_com(ofl, argc, argv, usage);
843 	else
844 		ret = process_files_com(ofl, argc, argv);
845 
846 	optind = ooptind;
847 	return (ret);
848 }
849 
850 /*
851  * Parse the items in a '-z guidance' value, and set the ofl_guideflags.
852  * A guidance option looks like this:
853  *
854  *	-z guidance[=item1,item2,...]
855  *
856  * Where each item specifies categories of guidance messages to suppress,
857  * each starting with the prefix 'no'. We allow arbitrary whitespace between
858  * the items, allow multiple ',' delimiters without an intervening item, and
859  * quietly ignore any items we don't recognize.
860  *
861  * -	Such items are likely to be known to newer versions of the linker,
862  *	and we do not want an older version of the linker to
863  *	complain about them.
864  *
865  * -	Times and standards can change, and so we wish to reserve the
866  *	right to make an old item that no longer makes sense go away.
867  *	Quietly ignoring unrecognized items facilitates this.
868  *
869  * However, we always display unrecognized items in debug output.
870  *
871  * entry:
872  *	ofl - Output descriptor
873  *	optarg - option string to be processed. This will either be a NULL
874  *		terminated 'guidance', or it will be 'guidance=' followed
875  *		by the item tokens as described above.
876  *
877  * exit:
878  *	Returns TRUE (1) on success, FALSE (0) on failure.
879  *
880  */
881 static Boolean
882 guidance_parse(Ofl_desc *ofl, char *optarg)
883 {
884 	typedef struct {
885 		const char	*name;
886 		ofl_guideflag_t	flag;
887 	} item_desc;
888 
889 	static  item_desc items[] = {
890 		{ MSG_ORIG(MSG_ARG_GUIDE_NO_ALL),	FLG_OFG_NO_ALL },
891 
892 		{ MSG_ORIG(MSG_ARG_GUIDE_NO_DEFS),	FLG_OFG_NO_DEFS },
893 		{ MSG_ORIG(MSG_ARG_GUIDE_NO_DIRECT),	FLG_OFG_NO_DB },
894 		{ MSG_ORIG(MSG_ARG_GUIDE_NO_LAZYLOAD),	FLG_OFG_NO_LAZY },
895 		{ MSG_ORIG(MSG_ARG_GUIDE_NO_MAPFILE),	FLG_OFG_NO_MF },
896 		{ MSG_ORIG(MSG_ARG_GUIDE_NO_TEXT),	FLG_OFG_NO_TEXT },
897 		{ MSG_ORIG(MSG_ARG_GUIDE_NO_UNUSED),	FLG_OFG_NO_UNUSED },
898 		{ NULL,					0 }
899 	};
900 
901 	char		*lasts, *name;
902 	item_desc	*item;
903 	ofl_guideflag_t	ofl_guideflags = FLG_OFG_ENABLE;
904 
905 	/*
906 	 * Skip the 'guidance' prefix. If NULL terminated, there are no
907 	 * item values to parse. Otherwise, skip the '=' and parse the items.
908 	 */
909 	optarg += MSG_ARG_GUIDE_SIZE;
910 	if (*optarg == '=') {
911 		optarg++;
912 
913 		if ((name = libld_malloc(strlen(optarg) + 1)) == NULL)
914 			return (FALSE);
915 		(void) strcpy(name, optarg);
916 
917 		if ((name = strtok_r(name, MSG_ORIG(MSG_ARG_GUIDE_DELIM),
918 		    &lasts)) != NULL) {
919 			do {
920 				for (item = items; item->name != NULL; item++)
921 					if (strcasecmp(name, item->name) == 0)
922 						break;
923 				if (item->name == NULL) {
924 					DBG_CALL(Dbg_args_guidance_unknown(
925 					    ofl->ofl_lml, name));
926 					continue;
927 				}
928 				ofl_guideflags |= item->flag;
929 			} while ((name = strtok_r(NULL,
930 			    MSG_ORIG(MSG_ARG_GUIDE_DELIM), &lasts)) != NULL);
931 		}
932 	}
933 
934 	/*
935 	 * If -zguidance is used more than once, we take the first one. We
936 	 * do this quietly if they have identical options, and with a warning
937 	 * otherwise.
938 	 */
939 	if ((initial_guidance_flags & FLG_OFG_ENABLE) &&
940 	    (ofl_guideflags != initial_guidance_flags)) {
941 		ld_eprintf(ofl, ERR_WARNING_NF, MSG_INTL(MSG_ARG_MTONCE),
942 		    MSG_ORIG(MSG_ARG_ZGUIDE));
943 		return (TRUE);
944 	}
945 
946 	/*
947 	 * First time: Save the flags for comparison to any subsequent
948 	 * -z guidance that comes along, and OR the resulting flags into
949 	 * the flags kept in the output descriptor.
950 	 */
951 	initial_guidance_flags = ofl_guideflags;
952 	ofl->ofl_guideflags |= ofl_guideflags;
953 	return (TRUE);
954 }
955 
956 /*
957  * Parse the -z assert-deflib option. This option can appear in two different
958  * forms:
959  *	-z assert-deflib
960  *	-z assert-deflib=libfred.so
961  *
962  * Either form enables this option, the latter form marks libfred.so as an
963  * exempt library from the check. It is valid to have multiple invocations of
964  * the second form. We silently ignore mulitple occurrences of the first form
965  * and multiple invocations of the first form when the second form also occurs.
966  *
967  * We only return false when we have an internal error, such as the failure of
968  * aplist_append. Every other time we return true, but we have the appropriate
969  * fatal flags set beacuse of the ld_eprintf.
970  */
971 static int
972 assdeflib_parse(Ofl_desc *ofl, char *optarg)
973 {
974 	size_t olen, mlen;
975 	ofl->ofl_flags |= FLG_OF_ADEFLIB;
976 
977 	olen = strlen(optarg);
978 	/* Minimum size of assert-deflib=lib%s.so */
979 	mlen = MSG_ARG_ASSDEFLIB_SIZE + 1 + MSG_STR_LIB_SIZE +
980 	    MSG_STR_SOEXT_SIZE;
981 	if (olen > MSG_ARG_ASSDEFLIB_SIZE) {
982 		if (optarg[MSG_ARG_ASSDEFLIB_SIZE] != '=') {
983 			ld_eprintf(ofl, ERR_FATAL, MSG_INTL(MSG_ARG_ILLEGAL),
984 			    MSG_ORIG(MSG_ARG_ASSDEFLIB), optarg);
985 			return (TRUE);
986 		}
987 
988 		if (strncmp(optarg + MSG_ARG_ASSDEFLIB_SIZE + 1,
989 		    MSG_ORIG(MSG_STR_LIB), MSG_STR_LIB_SIZE) != 0 ||
990 		    strcmp(optarg + olen - MSG_STR_SOEXT_SIZE,
991 		    MSG_ORIG(MSG_STR_SOEXT)) != 0 || olen <= mlen) {
992 			ld_eprintf(ofl, ERR_FATAL,
993 			    MSG_INTL(MSG_ARG_ASSDEFLIB_MALFORMED), optarg);
994 			return (TRUE);
995 		}
996 
997 		if (aplist_append(&ofl->ofl_assdeflib, optarg +
998 		    MSG_ARG_ASSDEFLIB_SIZE + 1, AL_CNT_ASSDEFLIB) == NULL)
999 			return (FALSE);
1000 	}
1001 
1002 	return (TRUE);
1003 }
1004 
1005 static int	optitle = 0;
1006 /*
1007  * Parsing options pass1 for process_flags().
1008  */
1009 static uintptr_t
1010 parseopt_pass1(Ofl_desc *ofl, int argc, char **argv, int *usage)
1011 {
1012 	int	c, ndx = optind;
1013 
1014 	/*
1015 	 * The -32, -64 and -ztarget options are special, in that we validate
1016 	 * them, but otherwise ignore them. libld.so (this code) is called
1017 	 * from the ld front end program. ld has already examined the
1018 	 * arguments to determine the output class and machine type of the
1019 	 * output object, as reflected in the version (32/64) of ld_main()
1020 	 * that was called and the value of the 'mach' argument passed.
1021 	 * By time execution reaches this point, these options have already
1022 	 * been seen and acted on.
1023 	 */
1024 	while ((c = ld_getopt(ofl->ofl_lml, ndx, argc, argv)) != -1) {
1025 
1026 		switch (c) {
1027 		case '3':
1028 			DBG_CALL(Dbg_args_option(ofl->ofl_lml, ndx, c, optarg));
1029 
1030 			/*
1031 			 * -32 is processed by ld to determine the output class.
1032 			 * Here we sanity check the option incase some other
1033 			 * -3* option is mistakenly passed to us.
1034 			 */
1035 			if (optarg[0] != '2')
1036 				ld_eprintf(ofl, ERR_FATAL,
1037 				    MSG_INTL(MSG_ARG_ILLEGAL),
1038 				    MSG_ORIG(MSG_ARG_3), optarg);
1039 			continue;
1040 
1041 		case '6':
1042 			DBG_CALL(Dbg_args_option(ofl->ofl_lml, ndx, c, optarg));
1043 
1044 			/*
1045 			 * -64 is processed by ld to determine the output class.
1046 			 * Here we sanity check the option incase some other
1047 			 * -6* option is mistakenly passed to us.
1048 			 */
1049 			if (optarg[0] != '4')
1050 				ld_eprintf(ofl, ERR_FATAL,
1051 				    MSG_INTL(MSG_ARG_ILLEGAL),
1052 				    MSG_ORIG(MSG_ARG_6), optarg);
1053 			continue;
1054 
1055 		case 'a':
1056 			DBG_CALL(Dbg_args_option(ofl->ofl_lml, ndx, c, NULL));
1057 			aflag = TRUE;
1058 			break;
1059 
1060 		case 'b':
1061 			DBG_CALL(Dbg_args_option(ofl->ofl_lml, ndx, c, NULL));
1062 			bflag = TRUE;
1063 
1064 			/*
1065 			 * This is a hack, and may be undone later.
1066 			 * The -b option is only used to build the Unix
1067 			 * kernel and its related kernel-mode modules.
1068 			 * We do not want those files to get a .SUNW_ldynsym
1069 			 * section. At least for now, the kernel makes no
1070 			 * use of .SUNW_ldynsym, and we do not want to use
1071 			 * the space to hold it. Therefore, we overload
1072 			 * the use of -b to also imply -znoldynsym.
1073 			 */
1074 			ofl->ofl_flags |= FLG_OF_NOLDYNSYM;
1075 			break;
1076 
1077 		case 'c':
1078 			DBG_CALL(Dbg_args_option(ofl->ofl_lml, ndx, c, optarg));
1079 			if (ofl->ofl_config)
1080 				ld_eprintf(ofl, ERR_WARNING_NF,
1081 				    MSG_INTL(MSG_ARG_MTONCE),
1082 				    MSG_ORIG(MSG_ARG_C));
1083 			else
1084 				ofl->ofl_config = optarg;
1085 			break;
1086 
1087 		case 'C':
1088 			DBG_CALL(Dbg_args_option(ofl->ofl_lml, ndx, c, NULL));
1089 			demangle_flag = 1;
1090 			break;
1091 
1092 		case 'd':
1093 			DBG_CALL(Dbg_args_option(ofl->ofl_lml, ndx, c, optarg));
1094 			if ((optarg[0] == 'n') && (optarg[1] == '\0')) {
1095 				if (dflag != SET_UNKNOWN)
1096 					ld_eprintf(ofl, ERR_WARNING_NF,
1097 					    MSG_INTL(MSG_ARG_MTONCE),
1098 					    MSG_ORIG(MSG_ARG_D));
1099 				else
1100 					dflag = SET_FALSE;
1101 			} else if ((optarg[0] == 'y') && (optarg[1] == '\0')) {
1102 				if (dflag != SET_UNKNOWN)
1103 					ld_eprintf(ofl, ERR_WARNING_NF,
1104 					    MSG_INTL(MSG_ARG_MTONCE),
1105 					    MSG_ORIG(MSG_ARG_D));
1106 				else
1107 					dflag = SET_TRUE;
1108 			} else {
1109 				ld_eprintf(ofl, ERR_FATAL,
1110 				    MSG_INTL(MSG_ARG_ILLEGAL),
1111 				    MSG_ORIG(MSG_ARG_D), optarg);
1112 			}
1113 			break;
1114 
1115 		case 'e':
1116 			DBG_CALL(Dbg_args_option(ofl->ofl_lml, ndx, c, optarg));
1117 			if (ofl->ofl_entry)
1118 				ld_eprintf(ofl, ERR_WARNING_NF,
1119 				    MSG_INTL(MSG_MARG_MTONCE),
1120 				    MSG_INTL(MSG_MARG_ENTRY));
1121 			else
1122 				ofl->ofl_entry = (void *)optarg;
1123 			break;
1124 
1125 		case 'f':
1126 			DBG_CALL(Dbg_args_option(ofl->ofl_lml, ndx, c, optarg));
1127 			if (ofl->ofl_filtees &&
1128 			    (!(ofl->ofl_flags & FLG_OF_AUX))) {
1129 				ld_eprintf(ofl, ERR_FATAL,
1130 				    MSG_INTL(MSG_MARG_INCOMP),
1131 				    MSG_INTL(MSG_MARG_FILTER_AUX),
1132 				    MSG_INTL(MSG_MARG_FILTER));
1133 			} else {
1134 				if ((ofl->ofl_filtees =
1135 				    add_string(ofl->ofl_filtees, optarg)) ==
1136 				    (const char *)S_ERROR)
1137 					return (S_ERROR);
1138 				ofl->ofl_flags |= FLG_OF_AUX;
1139 			}
1140 			break;
1141 
1142 		case 'F':
1143 			DBG_CALL(Dbg_args_option(ofl->ofl_lml, ndx, c, optarg));
1144 			if (ofl->ofl_filtees &&
1145 			    (ofl->ofl_flags & FLG_OF_AUX)) {
1146 				ld_eprintf(ofl, ERR_FATAL,
1147 				    MSG_INTL(MSG_MARG_INCOMP),
1148 				    MSG_INTL(MSG_MARG_FILTER),
1149 				    MSG_INTL(MSG_MARG_FILTER_AUX));
1150 			} else {
1151 				if ((ofl->ofl_filtees =
1152 				    add_string(ofl->ofl_filtees, optarg)) ==
1153 				    (const char *)S_ERROR)
1154 					return (S_ERROR);
1155 			}
1156 			break;
1157 
1158 		case 'h':
1159 			DBG_CALL(Dbg_args_option(ofl->ofl_lml, ndx, c, optarg));
1160 			if (ofl->ofl_soname)
1161 				ld_eprintf(ofl, ERR_WARNING_NF,
1162 				    MSG_INTL(MSG_MARG_MTONCE),
1163 				    MSG_INTL(MSG_MARG_SONAME));
1164 			else
1165 				ofl->ofl_soname = (const char *)optarg;
1166 			break;
1167 
1168 		case 'i':
1169 			DBG_CALL(Dbg_args_option(ofl->ofl_lml, ndx, c, NULL));
1170 			ofl->ofl_flags |= FLG_OF_IGNENV;
1171 			break;
1172 
1173 		case 'I':
1174 			DBG_CALL(Dbg_args_option(ofl->ofl_lml, ndx, c, optarg));
1175 			if (ofl->ofl_interp)
1176 				ld_eprintf(ofl, ERR_WARNING_NF,
1177 				    MSG_INTL(MSG_ARG_MTONCE),
1178 				    MSG_ORIG(MSG_ARG_CI));
1179 			else
1180 				ofl->ofl_interp = (const char *)optarg;
1181 			break;
1182 
1183 		case 'l':
1184 			DBG_CALL(Dbg_args_option(ofl->ofl_lml, ndx, c, optarg));
1185 			/*
1186 			 * For now, count any library as a shared object.  This
1187 			 * is used to size the internal symbol cache.  This
1188 			 * value is recalculated later on actual file processing
1189 			 * to get an accurate shared object count.
1190 			 */
1191 			ofl->ofl_soscnt++;
1192 			break;
1193 
1194 		case 'm':
1195 			DBG_CALL(Dbg_args_option(ofl->ofl_lml, ndx, c, NULL));
1196 			ofl->ofl_flags |= FLG_OF_GENMAP;
1197 			break;
1198 
1199 		case 'o':
1200 			DBG_CALL(Dbg_args_option(ofl->ofl_lml, ndx, c, optarg));
1201 			if (ofl->ofl_name)
1202 				ld_eprintf(ofl, ERR_WARNING_NF,
1203 				    MSG_INTL(MSG_MARG_MTONCE),
1204 				    MSG_INTL(MSG_MARG_OUTFILE));
1205 			else
1206 				ofl->ofl_name = (const char *)optarg;
1207 			break;
1208 
1209 		case 'p':
1210 			DBG_CALL(Dbg_args_option(ofl->ofl_lml, ndx, c, optarg));
1211 
1212 			/*
1213 			 * Multiple instances of this option may occur.  Each
1214 			 * additional instance is effectively concatenated to
1215 			 * the previous separated by a colon.
1216 			 */
1217 			if (*optarg != '\0') {
1218 				if ((ofl->ofl_audit =
1219 				    add_string(ofl->ofl_audit,
1220 				    optarg)) == (const char *)S_ERROR)
1221 					return (S_ERROR);
1222 			}
1223 			break;
1224 
1225 		case 'P':
1226 			DBG_CALL(Dbg_args_option(ofl->ofl_lml, ndx, c, optarg));
1227 
1228 			/*
1229 			 * Multiple instances of this option may occur.  Each
1230 			 * additional instance is effectively concatenated to
1231 			 * the previous separated by a colon.
1232 			 */
1233 			if (*optarg != '\0') {
1234 				if ((ofl->ofl_depaudit =
1235 				    add_string(ofl->ofl_depaudit,
1236 				    optarg)) == (const char *)S_ERROR)
1237 					return (S_ERROR);
1238 			}
1239 			break;
1240 
1241 		case 'r':
1242 			DBG_CALL(Dbg_args_option(ofl->ofl_lml, ndx, c, NULL));
1243 			otype = OT_RELOC;
1244 			break;
1245 
1246 		case 'R':
1247 			DBG_CALL(Dbg_args_option(ofl->ofl_lml, ndx, c, optarg));
1248 
1249 			/*
1250 			 * Multiple instances of this option may occur.  Each
1251 			 * additional instance is effectively concatenated to
1252 			 * the previous separated by a colon.
1253 			 */
1254 			if (*optarg != '\0') {
1255 				if ((ofl->ofl_rpath =
1256 				    add_string(ofl->ofl_rpath,
1257 				    optarg)) == (const char *)S_ERROR)
1258 					return (S_ERROR);
1259 			}
1260 			break;
1261 
1262 		case 's':
1263 			DBG_CALL(Dbg_args_option(ofl->ofl_lml, ndx, c, NULL));
1264 			sflag = TRUE;
1265 			break;
1266 
1267 		case 't':
1268 			DBG_CALL(Dbg_args_option(ofl->ofl_lml, ndx, c, NULL));
1269 			ofl->ofl_flags |= FLG_OF_NOWARN;
1270 			break;
1271 
1272 		case 'u':
1273 			DBG_CALL(Dbg_args_option(ofl->ofl_lml, ndx, c, optarg));
1274 			break;
1275 
1276 		case 'z':
1277 			DBG_CALL(Dbg_args_option(ofl->ofl_lml, ndx, c, optarg));
1278 
1279 			/*
1280 			 * Skip comma that might be present between -z and its
1281 			 * argument (e.g. if -Wl,-z,assert-deflib was passed).
1282 			 */
1283 			if (strncmp(optarg, MSG_ORIG(MSG_STR_COMMA),
1284 			    MSG_STR_COMMA_SIZE) == 0)
1285 				optarg++;
1286 
1287 			/*
1288 			 * For specific help, print our usage message and exit
1289 			 * immediately to ensure a 0 return code.
1290 			 */
1291 			if (strncmp(optarg, MSG_ORIG(MSG_ARG_HELP),
1292 			    MSG_ARG_HELP_SIZE) == 0) {
1293 				usage_mesg(TRUE);
1294 				exit(0);
1295 			}
1296 
1297 			/*
1298 			 * For some options set a flag - further consistancy
1299 			 * checks will be carried out in check_flags().
1300 			 */
1301 			if ((strncmp(optarg, MSG_ORIG(MSG_ARG_LD32),
1302 			    MSG_ARG_LD32_SIZE) == 0) ||
1303 			    (strncmp(optarg, MSG_ORIG(MSG_ARG_LD64),
1304 			    MSG_ARG_LD64_SIZE) == 0)) {
1305 				if (createargv(ofl, usage) == S_ERROR)
1306 					return (S_ERROR);
1307 
1308 			} else if (
1309 			    strcmp(optarg, MSG_ORIG(MSG_ARG_DEFS)) == 0) {
1310 				if (zdflag != SET_UNKNOWN)
1311 					ld_eprintf(ofl, ERR_WARNING_NF,
1312 					    MSG_INTL(MSG_ARG_MTONCE),
1313 					    MSG_ORIG(MSG_ARG_ZDEFNODEF));
1314 				else
1315 					zdflag = SET_TRUE;
1316 				ofl->ofl_guideflags |= FLG_OFG_NO_DEFS;
1317 			} else if (strcmp(optarg,
1318 			    MSG_ORIG(MSG_ARG_NODEFS)) == 0) {
1319 				if (zdflag != SET_UNKNOWN)
1320 					ld_eprintf(ofl, ERR_WARNING_NF,
1321 					    MSG_INTL(MSG_ARG_MTONCE),
1322 					    MSG_ORIG(MSG_ARG_ZDEFNODEF));
1323 				else
1324 					zdflag = SET_FALSE;
1325 				ofl->ofl_guideflags |= FLG_OFG_NO_DEFS;
1326 			} else if (strcmp(optarg,
1327 			    MSG_ORIG(MSG_ARG_TEXT)) == 0) {
1328 				if (ztflag &&
1329 				    (ztflag != MSG_ORIG(MSG_ARG_ZTEXT)))
1330 					ld_eprintf(ofl, ERR_FATAL,
1331 					    MSG_INTL(MSG_ARG_INCOMP),
1332 					    MSG_ORIG(MSG_ARG_ZTEXT),
1333 					    ztflag);
1334 				ztflag = MSG_ORIG(MSG_ARG_ZTEXT);
1335 			} else if (strcmp(optarg,
1336 			    MSG_ORIG(MSG_ARG_TEXTOFF)) == 0) {
1337 				if (ztflag &&
1338 				    (ztflag != MSG_ORIG(MSG_ARG_ZTEXTOFF)))
1339 					ld_eprintf(ofl, ERR_FATAL,
1340 					    MSG_INTL(MSG_ARG_INCOMP),
1341 					    MSG_ORIG(MSG_ARG_ZTEXTOFF),
1342 					    ztflag);
1343 				ztflag = MSG_ORIG(MSG_ARG_ZTEXTOFF);
1344 			} else if (strcmp(optarg,
1345 			    MSG_ORIG(MSG_ARG_TEXTWARN)) == 0) {
1346 				if (ztflag &&
1347 				    (ztflag != MSG_ORIG(MSG_ARG_ZTEXTWARN)))
1348 					ld_eprintf(ofl, ERR_FATAL,
1349 					    MSG_INTL(MSG_ARG_INCOMP),
1350 					    MSG_ORIG(MSG_ARG_ZTEXTWARN),
1351 					    ztflag);
1352 				ztflag = MSG_ORIG(MSG_ARG_ZTEXTWARN);
1353 
1354 			/*
1355 			 * For other options simply set the ofl flags directly.
1356 			 */
1357 			} else if (strcmp(optarg,
1358 			    MSG_ORIG(MSG_ARG_RESCAN)) == 0) {
1359 				ofl->ofl_flags1 |= FLG_OF1_RESCAN;
1360 			} else if (strcmp(optarg,
1361 			    MSG_ORIG(MSG_ARG_ABSEXEC)) == 0) {
1362 				ofl->ofl_flags1 |= FLG_OF1_ABSEXEC;
1363 			} else if (strcmp(optarg,
1364 			    MSG_ORIG(MSG_ARG_LOADFLTR)) == 0) {
1365 				zlflag = TRUE;
1366 			} else if (strcmp(optarg,
1367 			    MSG_ORIG(MSG_ARG_NORELOC)) == 0) {
1368 				ofl->ofl_dtflags_1 |= DF_1_NORELOC;
1369 			} else if (strcmp(optarg,
1370 			    MSG_ORIG(MSG_ARG_NOVERSION)) == 0) {
1371 				ofl->ofl_flags |= FLG_OF_NOVERSEC;
1372 			} else if (strcmp(optarg,
1373 			    MSG_ORIG(MSG_ARG_MULDEFS)) == 0) {
1374 				ofl->ofl_flags |= FLG_OF_MULDEFS;
1375 			} else if (strcmp(optarg,
1376 			    MSG_ORIG(MSG_ARG_REDLOCSYM)) == 0) {
1377 				ofl->ofl_flags |= FLG_OF_REDLSYM;
1378 			} else if (strcmp(optarg,
1379 			    MSG_ORIG(MSG_ARG_INITFIRST)) == 0) {
1380 				ofl->ofl_dtflags_1 |= DF_1_INITFIRST;
1381 			} else if (strcmp(optarg,
1382 			    MSG_ORIG(MSG_ARG_NODELETE)) == 0) {
1383 				ofl->ofl_dtflags_1 |= DF_1_NODELETE;
1384 			} else if (strcmp(optarg,
1385 			    MSG_ORIG(MSG_ARG_NOPARTIAL)) == 0) {
1386 				ofl->ofl_flags1 |= FLG_OF1_NOPARTI;
1387 			} else if (strcmp(optarg,
1388 			    MSG_ORIG(MSG_ARG_NOOPEN)) == 0) {
1389 				ofl->ofl_dtflags_1 |= DF_1_NOOPEN;
1390 			} else if (strcmp(optarg,
1391 			    MSG_ORIG(MSG_ARG_NOW)) == 0) {
1392 				ofl->ofl_dtflags_1 |= DF_1_NOW;
1393 				ofl->ofl_dtflags |= DF_BIND_NOW;
1394 			} else if (strcmp(optarg,
1395 			    MSG_ORIG(MSG_ARG_ORIGIN)) == 0) {
1396 				ofl->ofl_dtflags_1 |= DF_1_ORIGIN;
1397 				ofl->ofl_dtflags |= DF_ORIGIN;
1398 			} else if (strcmp(optarg,
1399 			    MSG_ORIG(MSG_ARG_NODEFAULTLIB)) == 0) {
1400 				ofl->ofl_dtflags_1 |= DF_1_NODEFLIB;
1401 			} else if (strcmp(optarg,
1402 			    MSG_ORIG(MSG_ARG_NODUMP)) == 0) {
1403 				ofl->ofl_dtflags_1 |= DF_1_NODUMP;
1404 			} else if (strcmp(optarg,
1405 			    MSG_ORIG(MSG_ARG_ENDFILTEE)) == 0) {
1406 				ofl->ofl_dtflags_1 |= DF_1_ENDFILTEE;
1407 			} else if (strcmp(optarg,
1408 			    MSG_ORIG(MSG_ARG_VERBOSE)) == 0) {
1409 				ofl->ofl_flags |= FLG_OF_VERBOSE;
1410 			} else if (strcmp(optarg,
1411 			    MSG_ORIG(MSG_ARG_COMBRELOC)) == 0) {
1412 				ofl->ofl_flags |= FLG_OF_COMREL;
1413 			} else if (strcmp(optarg,
1414 			    MSG_ORIG(MSG_ARG_NOCOMBRELOC)) == 0) {
1415 				ofl->ofl_flags |= FLG_OF_NOCOMREL;
1416 			} else if (strcmp(optarg,
1417 			    MSG_ORIG(MSG_ARG_NOCOMPSTRTAB)) == 0) {
1418 				ofl->ofl_flags1 |= FLG_OF1_NCSTTAB;
1419 			} else if (strcmp(optarg,
1420 			    MSG_ORIG(MSG_ARG_NOINTERP)) == 0) {
1421 				ofl->ofl_flags1 |= FLG_OF1_NOINTRP;
1422 			} else if (strcmp(optarg,
1423 			    MSG_ORIG(MSG_ARG_INTERPOSE)) == 0) {
1424 				zinflag = TRUE;
1425 			} else if (strcmp(optarg,
1426 			    MSG_ORIG(MSG_ARG_IGNORE)) == 0) {
1427 				ofl->ofl_flags1 |= FLG_OF1_IGNPRC;
1428 			} else if (strcmp(optarg,
1429 			    MSG_ORIG(MSG_ARG_RELAXRELOC)) == 0) {
1430 				ofl->ofl_flags1 |= FLG_OF1_RLXREL;
1431 			} else if (strcmp(optarg,
1432 			    MSG_ORIG(MSG_ARG_NORELAXRELOC)) == 0) {
1433 				ofl->ofl_flags1 |= FLG_OF1_NRLXREL;
1434 			} else if (strcmp(optarg,
1435 			    MSG_ORIG(MSG_ARG_NOLDYNSYM)) == 0) {
1436 				ofl->ofl_flags |= FLG_OF_NOLDYNSYM;
1437 			} else if (strcmp(optarg,
1438 			    MSG_ORIG(MSG_ARG_GLOBAUDIT)) == 0) {
1439 				ofl->ofl_dtflags_1 |= DF_1_GLOBAUDIT;
1440 			} else if (strcmp(optarg,
1441 			    MSG_ORIG(MSG_ARG_NOSIGHANDLER)) == 0) {
1442 				ofl->ofl_flags1 |= FLG_OF1_NOSGHND;
1443 			} else if (strcmp(optarg,
1444 			    MSG_ORIG(MSG_ARG_SYMBOLCAP)) == 0) {
1445 				ofl->ofl_flags |= FLG_OF_OTOSCAP;
1446 
1447 			/*
1448 			 * Check archive group usage
1449 			 *	-z rescan-start ... -z rescan-end
1450 			 * to ensure they don't overlap and are well formed.
1451 			 */
1452 			} else if (strcmp(optarg,
1453 			    MSG_ORIG(MSG_ARG_RESCAN_START)) == 0) {
1454 				if (ofl->ofl_ars_gsandx == 0) {
1455 					ofl->ofl_ars_gsandx = ndx;
1456 				} else if (ofl->ofl_ars_gsandx > 0) {
1457 					/* Another group is still open */
1458 					ld_eprintf(ofl, ERR_FATAL,
1459 					    MSG_INTL(MSG_ARG_AR_GRP_OLAP),
1460 					    MSG_INTL(MSG_MARG_AR_GRPS));
1461 					/* Don't report cascading errors */
1462 					ofl->ofl_ars_gsandx = -1;
1463 				}
1464 			} else if (strcmp(optarg,
1465 			    MSG_ORIG(MSG_ARG_RESCAN_END)) == 0) {
1466 				if (ofl->ofl_ars_gsandx > 0) {
1467 					ofl->ofl_ars_gsandx = 0;
1468 				} else if (ofl->ofl_ars_gsandx == 0) {
1469 					/* There was no matching begin */
1470 					ld_eprintf(ofl, ERR_FATAL,
1471 					    MSG_INTL(MSG_ARG_AR_GRP_BAD),
1472 					    MSG_INTL(MSG_MARG_AR_GRP_END),
1473 					    MSG_INTL(MSG_MARG_AR_GRP_START));
1474 					/* Don't report cascading errors */
1475 					ofl->ofl_ars_gsandx = -1;
1476 				}
1477 
1478 			/*
1479 			 * If -z wrap is seen, enter the symbol to be wrapped
1480 			 * into the wrap AVL tree.
1481 			 */
1482 			} else if (strncmp(optarg, MSG_ORIG(MSG_ARG_WRAP),
1483 			    MSG_ARG_WRAP_SIZE) == 0) {
1484 				if (ld_wrap_enter(ofl,
1485 				    optarg + MSG_ARG_WRAP_SIZE) == NULL)
1486 					return (S_ERROR);
1487 			} else if (strncmp(optarg, MSG_ORIG(MSG_ARG_ASLR),
1488 			    MSG_ARG_ASLR_SIZE) == 0) {
1489 				char *p = optarg + MSG_ARG_ASLR_SIZE;
1490 				if (*p == '\0') {
1491 					ofl->ofl_aslr = 1;
1492 				} else if (*p == '=') {
1493 					p++;
1494 
1495 					if ((strcmp(p,
1496 					    MSG_ORIG(MSG_ARG_ENABLED)) == 0) ||
1497 					    (strcmp(p,
1498 					    MSG_ORIG(MSG_ARG_ENABLE)) == 0)) {
1499 						ofl->ofl_aslr = 1;
1500 					} else if ((strcmp(p,
1501 					    MSG_ORIG(MSG_ARG_DISABLED)) == 0) ||
1502 					    (strcmp(p,
1503 					    MSG_ORIG(MSG_ARG_DISABLE)) == 0)) {
1504 						ofl->ofl_aslr = -1;
1505 					} else {
1506 						ld_eprintf(ofl, ERR_FATAL,
1507 						    MSG_INTL(MSG_ARG_ILLEGAL),
1508 						    MSG_ORIG(MSG_ARG_ZASLR), p);
1509 						return (S_ERROR);
1510 					}
1511 				} else {
1512 					ld_eprintf(ofl, ERR_FATAL,
1513 					    MSG_INTL(MSG_ARG_ILLEGAL),
1514 					    MSG_ORIG(MSG_ARG_Z), optarg);
1515 					return (S_ERROR);
1516 				}
1517 			} else if ((strncmp(optarg, MSG_ORIG(MSG_ARG_GUIDE),
1518 			    MSG_ARG_GUIDE_SIZE) == 0) &&
1519 			    ((optarg[MSG_ARG_GUIDE_SIZE] == '=') ||
1520 			    (optarg[MSG_ARG_GUIDE_SIZE] == '\0'))) {
1521 				if (!guidance_parse(ofl, optarg))
1522 					return (S_ERROR);
1523 			} else if (strcmp(optarg,
1524 			    MSG_ORIG(MSG_ARG_FATWARN)) == 0) {
1525 				if (zfwflag  == SET_FALSE) {
1526 					ld_eprintf(ofl, ERR_WARNING_NF,
1527 					    MSG_INTL(MSG_ARG_MTONCE),
1528 					    MSG_ORIG(MSG_ARG_ZFATWNOFATW));
1529 				} else {
1530 					zfwflag = SET_TRUE;
1531 					ofl->ofl_flags |= FLG_OF_FATWARN;
1532 				}
1533 			} else if (strcmp(optarg,
1534 			    MSG_ORIG(MSG_ARG_NOFATWARN)) == 0) {
1535 				if (zfwflag  == SET_TRUE)
1536 					ld_eprintf(ofl, ERR_WARNING_NF,
1537 					    MSG_INTL(MSG_ARG_MTONCE),
1538 					    MSG_ORIG(MSG_ARG_ZFATWNOFATW));
1539 				else
1540 					zfwflag = SET_FALSE;
1541 
1542 			/*
1543 			 * Process everything related to -z assert-deflib. This
1544 			 * must be done in pass 1 because it gets used in pass
1545 			 * 2.
1546 			 */
1547 			} else if (strncmp(optarg, MSG_ORIG(MSG_ARG_ASSDEFLIB),
1548 			    MSG_ARG_ASSDEFLIB_SIZE) == 0) {
1549 				if (assdeflib_parse(ofl, optarg) != TRUE)
1550 					return (S_ERROR);
1551 
1552 			/*
1553 			 * Process new-style output type specification, which
1554 			 * we'll use in pass 2 and throughout.
1555 			 */
1556 			} else if (strncmp(optarg, MSG_ORIG(MSG_ARG_TYPE),
1557 			    MSG_ARG_TYPE_SIZE) == 0) {
1558 				char *p = optarg + MSG_ARG_TYPE_SIZE;
1559 				if (*p != '=') {
1560 					ld_eprintf(ofl, ERR_FATAL,
1561 					    MSG_INTL(MSG_ARG_ILLEGAL),
1562 					    MSG_ORIG(MSG_ARG_Z), optarg);
1563 					return (S_ERROR);
1564 				}
1565 
1566 				p++;
1567 				if (strcmp(p,
1568 				    MSG_ORIG(MSG_ARG_TYPE_RELOC)) == 0) {
1569 					otype = OT_RELOC;
1570 				} else if (strcmp(p,
1571 				    MSG_ORIG(MSG_ARG_TYPE_EXEC)) == 0) {
1572 					otype = OT_EXEC;
1573 				} else if (strcmp(p,
1574 				    MSG_ORIG(MSG_ARG_TYPE_SHARED)) == 0) {
1575 					otype = OT_SHARED;
1576 				} else if (strcmp(p,
1577 				    MSG_ORIG(MSG_ARG_TYPE_KMOD)) == 0) {
1578 					otype = OT_KMOD;
1579 				} else {
1580 					ld_eprintf(ofl, ERR_FATAL,
1581 					    MSG_INTL(MSG_ARG_ILLEGAL),
1582 					    MSG_ORIG(MSG_ARG_Z), optarg);
1583 					return (S_ERROR);
1584 				}
1585 			/*
1586 			 * The following options just need validation as they
1587 			 * are interpreted on the second pass through the
1588 			 * command line arguments.
1589 			 */
1590 			} else if (
1591 			    strncmp(optarg, MSG_ORIG(MSG_ARG_INITARRAY),
1592 			    MSG_ARG_INITARRAY_SIZE) &&
1593 			    strncmp(optarg, MSG_ORIG(MSG_ARG_FINIARRAY),
1594 			    MSG_ARG_FINIARRAY_SIZE) &&
1595 			    strncmp(optarg, MSG_ORIG(MSG_ARG_PREINITARRAY),
1596 			    MSG_ARG_PREINITARRAY_SIZE) &&
1597 			    strncmp(optarg, MSG_ORIG(MSG_ARG_RTLDINFO),
1598 			    MSG_ARG_RTLDINFO_SIZE) &&
1599 			    strncmp(optarg, MSG_ORIG(MSG_ARG_DTRACE),
1600 			    MSG_ARG_DTRACE_SIZE) &&
1601 			    strcmp(optarg, MSG_ORIG(MSG_ARG_ALLEXTRT)) &&
1602 			    strcmp(optarg, MSG_ORIG(MSG_ARG_DFLEXTRT)) &&
1603 			    strcmp(optarg, MSG_ORIG(MSG_ARG_DIRECT)) &&
1604 			    strcmp(optarg, MSG_ORIG(MSG_ARG_NODIRECT)) &&
1605 			    strcmp(optarg, MSG_ORIG(MSG_ARG_GROUPPERM)) &&
1606 			    strcmp(optarg, MSG_ORIG(MSG_ARG_LAZYLOAD)) &&
1607 			    strcmp(optarg, MSG_ORIG(MSG_ARG_NOGROUPPERM)) &&
1608 			    strcmp(optarg, MSG_ORIG(MSG_ARG_NOLAZYLOAD)) &&
1609 			    strcmp(optarg, MSG_ORIG(MSG_ARG_NODEFERRED)) &&
1610 			    strcmp(optarg, MSG_ORIG(MSG_ARG_RECORD)) &&
1611 			    strcmp(optarg, MSG_ORIG(MSG_ARG_ALTEXEC64)) &&
1612 			    strcmp(optarg, MSG_ORIG(MSG_ARG_WEAKEXT)) &&
1613 			    strncmp(optarg, MSG_ORIG(MSG_ARG_TARGET),
1614 			    MSG_ARG_TARGET_SIZE) &&
1615 			    strcmp(optarg, MSG_ORIG(MSG_ARG_RESCAN_NOW)) &&
1616 			    strcmp(optarg, MSG_ORIG(MSG_ARG_DEFERRED))) {
1617 				ld_eprintf(ofl, ERR_FATAL,
1618 				    MSG_INTL(MSG_ARG_ILLEGAL),
1619 				    MSG_ORIG(MSG_ARG_Z), optarg);
1620 			}
1621 
1622 			break;
1623 
1624 		case 'D':
1625 			/*
1626 			 * If we have not yet read any input files go ahead
1627 			 * and process any debugging options (this allows any
1628 			 * argument processing, entrance criteria and library
1629 			 * initialization to be displayed).  Otherwise, if an
1630 			 * input file has been seen, skip interpretation until
1631 			 * process_files (this allows debugging to be turned
1632 			 * on and off around individual groups of files).
1633 			 */
1634 			Dflag = 1;
1635 			if (ofl->ofl_objscnt == 0) {
1636 				if (dbg_setup(ofl, optarg, 2) == 0)
1637 					return (S_ERROR);
1638 			}
1639 
1640 			/*
1641 			 * A diagnostic can only be provided after dbg_setup().
1642 			 * As this is the first diagnostic that can be produced
1643 			 * by ld(1), issue a title for timing and basic output.
1644 			 */
1645 			if ((optitle == 0) && DBG_ENABLED) {
1646 				optitle++;
1647 				DBG_CALL(Dbg_basic_options(ofl->ofl_lml));
1648 			}
1649 			DBG_CALL(Dbg_args_option(ofl->ofl_lml, ndx, c, optarg));
1650 			break;
1651 
1652 		case 'B':
1653 			DBG_CALL(Dbg_args_option(ofl->ofl_lml, ndx, c, optarg));
1654 			if (strcmp(optarg, MSG_ORIG(MSG_ARG_DIRECT)) == 0) {
1655 				if (Bdflag == SET_FALSE) {
1656 					ld_eprintf(ofl, ERR_FATAL,
1657 					    MSG_INTL(MSG_ARG_INCOMP),
1658 					    MSG_ORIG(MSG_ARG_BNODIRECT),
1659 					    MSG_ORIG(MSG_ARG_BDIRECT));
1660 				} else {
1661 					Bdflag = SET_TRUE;
1662 					ofl->ofl_guideflags |= FLG_OFG_NO_DB;
1663 				}
1664 			} else if (strcmp(optarg,
1665 			    MSG_ORIG(MSG_ARG_NODIRECT)) == 0) {
1666 				if (Bdflag == SET_TRUE) {
1667 					ld_eprintf(ofl, ERR_FATAL,
1668 					    MSG_INTL(MSG_ARG_INCOMP),
1669 					    MSG_ORIG(MSG_ARG_BDIRECT),
1670 					    MSG_ORIG(MSG_ARG_BNODIRECT));
1671 				} else {
1672 					Bdflag = SET_FALSE;
1673 					ofl->ofl_guideflags |= FLG_OFG_NO_DB;
1674 				}
1675 			} else if (strcmp(optarg,
1676 			    MSG_ORIG(MSG_STR_SYMBOLIC)) == 0)
1677 				Bsflag = TRUE;
1678 			else if (strcmp(optarg, MSG_ORIG(MSG_ARG_REDUCE)) == 0)
1679 				ofl->ofl_flags |= FLG_OF_PROCRED;
1680 			else if (strcmp(optarg, MSG_ORIG(MSG_STR_LOCAL)) == 0)
1681 				Blflag = TRUE;
1682 			else if (strcmp(optarg, MSG_ORIG(MSG_ARG_GROUP)) == 0)
1683 				Bgflag = TRUE;
1684 			else if (strcmp(optarg,
1685 			    MSG_ORIG(MSG_STR_ELIMINATE)) == 0)
1686 				Beflag = TRUE;
1687 			else if (strcmp(optarg,
1688 			    MSG_ORIG(MSG_ARG_TRANSLATOR)) == 0) {
1689 				ld_eprintf(ofl, ERR_WARNING,
1690 				    MSG_INTL(MSG_ARG_UNSUPPORTED),
1691 				    MSG_ORIG(MSG_ARG_BTRANSLATOR));
1692 			} else if (strcmp(optarg,
1693 			    MSG_ORIG(MSG_STR_LD_DYNAMIC)) &&
1694 			    strcmp(optarg, MSG_ORIG(MSG_ARG_STATIC))) {
1695 				ld_eprintf(ofl, ERR_FATAL,
1696 				    MSG_INTL(MSG_ARG_ILLEGAL),
1697 				    MSG_ORIG(MSG_ARG_CB), optarg);
1698 			}
1699 			break;
1700 
1701 		case 'G':
1702 			DBG_CALL(Dbg_args_option(ofl->ofl_lml, ndx, c, NULL));
1703 			otype = OT_SHARED;
1704 			break;
1705 
1706 		case 'L':
1707 			DBG_CALL(Dbg_args_option(ofl->ofl_lml, ndx, c, optarg));
1708 			break;
1709 
1710 		case 'M':
1711 			DBG_CALL(Dbg_args_option(ofl->ofl_lml, ndx, c, optarg));
1712 			if (aplist_append(&(ofl->ofl_maps), optarg,
1713 			    AL_CNT_OFL_MAPFILES) == NULL)
1714 				return (S_ERROR);
1715 			break;
1716 
1717 		case 'N':
1718 			DBG_CALL(Dbg_args_option(ofl->ofl_lml, ndx, c, optarg));
1719 			break;
1720 
1721 		case 'Q':
1722 			DBG_CALL(Dbg_args_option(ofl->ofl_lml, ndx, c, optarg));
1723 			if ((optarg[0] == 'n') && (optarg[1] == '\0')) {
1724 				if (Qflag != SET_UNKNOWN)
1725 					ld_eprintf(ofl, ERR_WARNING_NF,
1726 					    MSG_INTL(MSG_ARG_MTONCE),
1727 					    MSG_ORIG(MSG_ARG_CQ));
1728 				else
1729 					Qflag = SET_FALSE;
1730 			} else if ((optarg[0] == 'y') && (optarg[1] == '\0')) {
1731 				if (Qflag != SET_UNKNOWN)
1732 					ld_eprintf(ofl, ERR_WARNING_NF,
1733 					    MSG_INTL(MSG_ARG_MTONCE),
1734 					    MSG_ORIG(MSG_ARG_CQ));
1735 				else
1736 					Qflag = SET_TRUE;
1737 			} else {
1738 				ld_eprintf(ofl, ERR_FATAL,
1739 				    MSG_INTL(MSG_ARG_ILLEGAL),
1740 				    MSG_ORIG(MSG_ARG_CQ), optarg);
1741 			}
1742 			break;
1743 
1744 		case 'S':
1745 			DBG_CALL(Dbg_args_option(ofl->ofl_lml, ndx, c, optarg));
1746 			if (aplist_append(&lib_support, optarg,
1747 			    AL_CNT_SUPPORT) == NULL)
1748 				return (S_ERROR);
1749 			break;
1750 
1751 		case 'V':
1752 			DBG_CALL(Dbg_args_option(ofl->ofl_lml, ndx, c, NULL));
1753 			if (!Vflag)
1754 				(void) fprintf(stderr, MSG_ORIG(MSG_STR_STRNL),
1755 				    ofl->ofl_sgsid);
1756 			Vflag = TRUE;
1757 			break;
1758 
1759 		case 'Y':
1760 			DBG_CALL(Dbg_args_option(ofl->ofl_lml, ndx, c, optarg));
1761 			if (strncmp(optarg, MSG_ORIG(MSG_ARG_LCOM), 2) == 0) {
1762 				if (Llibdir)
1763 					ld_eprintf(ofl, ERR_WARNING_NF,
1764 					    MSG_INTL(MSG_ARG_MTONCE),
1765 					    MSG_ORIG(MSG_ARG_CYL));
1766 				else
1767 					Llibdir = optarg + 2;
1768 			} else if (strncmp(optarg,
1769 			    MSG_ORIG(MSG_ARG_UCOM), 2) == 0) {
1770 				if (Ulibdir)
1771 					ld_eprintf(ofl, ERR_WARNING_NF,
1772 					    MSG_INTL(MSG_ARG_MTONCE),
1773 					    MSG_ORIG(MSG_ARG_CYU));
1774 				else
1775 					Ulibdir = optarg + 2;
1776 			} else if (strncmp(optarg,
1777 			    MSG_ORIG(MSG_ARG_PCOM), 2) == 0) {
1778 				if (Plibpath)
1779 					ld_eprintf(ofl, ERR_WARNING_NF,
1780 					    MSG_INTL(MSG_ARG_MTONCE),
1781 					    MSG_ORIG(MSG_ARG_CYP));
1782 				else
1783 					Plibpath = optarg + 2;
1784 			} else {
1785 				ld_eprintf(ofl, ERR_FATAL,
1786 				    MSG_INTL(MSG_ARG_ILLEGAL),
1787 				    MSG_ORIG(MSG_ARG_CY), optarg);
1788 			}
1789 			break;
1790 
1791 		case '?':
1792 			DBG_CALL(Dbg_args_option(ofl->ofl_lml, ndx, c, NULL));
1793 			/*
1794 			 * If the option character is '-', we're looking at a
1795 			 * long option which couldn't be translated, display a
1796 			 * more useful error.
1797 			 */
1798 			if (optopt == '-') {
1799 				eprintf(ofl->ofl_lml, ERR_FATAL,
1800 				    MSG_INTL(MSG_ARG_LONG_UNKNOWN),
1801 				    argv[optind-1]);
1802 			} else {
1803 				eprintf(ofl->ofl_lml, ERR_FATAL,
1804 				    MSG_INTL(MSG_ARG_UNKNOWN), optopt);
1805 			}
1806 			(*usage)++;
1807 			break;
1808 
1809 		default:
1810 			break;
1811 		}
1812 
1813 		/*
1814 		 * Update the argument index for the next getopt() iteration.
1815 		 */
1816 		ndx = optind;
1817 	}
1818 	return (1);
1819 }
1820 
1821 /*
1822  * Parsing options pass2 for
1823  */
1824 static uintptr_t
1825 parseopt_pass2(Ofl_desc *ofl, int argc, char **argv)
1826 {
1827 	int	c, ndx = optind;
1828 
1829 	while ((c = ld_getopt(ofl->ofl_lml, ndx, argc, argv)) != -1) {
1830 		Ifl_desc	*ifl;
1831 		Sym_desc	*sdp;
1832 
1833 		switch (c) {
1834 			case 'l':
1835 				DBG_CALL(Dbg_args_option(ofl->ofl_lml, ndx, c,
1836 				    optarg));
1837 				if (ld_find_library(optarg, ofl) == S_ERROR)
1838 					return (S_ERROR);
1839 				break;
1840 			case 'B':
1841 				DBG_CALL(Dbg_args_option(ofl->ofl_lml, ndx, c,
1842 				    optarg));
1843 				if (strcmp(optarg,
1844 				    MSG_ORIG(MSG_STR_LD_DYNAMIC)) == 0) {
1845 					if (ofl->ofl_flags & FLG_OF_DYNAMIC)
1846 						ofl->ofl_flags |=
1847 						    FLG_OF_DYNLIBS;
1848 					else {
1849 						ld_eprintf(ofl, ERR_FATAL,
1850 						    MSG_INTL(MSG_ARG_ST_INCOMP),
1851 						    MSG_ORIG(MSG_ARG_BDYNAMIC));
1852 					}
1853 				} else if (strcmp(optarg,
1854 				    MSG_ORIG(MSG_ARG_STATIC)) == 0)
1855 					ofl->ofl_flags &= ~FLG_OF_DYNLIBS;
1856 				break;
1857 			case 'L':
1858 				DBG_CALL(Dbg_args_option(ofl->ofl_lml, ndx, c,
1859 				    optarg));
1860 				if (ld_add_libdir(ofl, optarg) == S_ERROR)
1861 					return (S_ERROR);
1862 				break;
1863 			case 'N':
1864 				DBG_CALL(Dbg_args_option(ofl->ofl_lml, ndx, c,
1865 				    optarg));
1866 				/*
1867 				 * Record DT_NEEDED string
1868 				 */
1869 				if (!(ofl->ofl_flags & FLG_OF_DYNAMIC))
1870 					ld_eprintf(ofl, ERR_FATAL,
1871 					    MSG_INTL(MSG_ARG_ST_INCOMP),
1872 					    MSG_ORIG(MSG_ARG_CN));
1873 				if (((ifl = libld_calloc(1,
1874 				    sizeof (Ifl_desc))) == NULL) ||
1875 				    (aplist_append(&ofl->ofl_sos, ifl,
1876 				    AL_CNT_OFL_LIBS) == NULL))
1877 					return (S_ERROR);
1878 
1879 				ifl->ifl_name = MSG_INTL(MSG_STR_COMMAND);
1880 				ifl->ifl_soname = optarg;
1881 				ifl->ifl_flags = (FLG_IF_NEEDSTR |
1882 				    FLG_IF_FILEREF | FLG_IF_DEPREQD);
1883 
1884 				break;
1885 			case 'D':
1886 				DBG_CALL(Dbg_args_option(ofl->ofl_lml, ndx, c,
1887 				    optarg));
1888 				(void) dbg_setup(ofl, optarg, 3);
1889 				break;
1890 			case 'u':
1891 				DBG_CALL(Dbg_args_option(ofl->ofl_lml, ndx, c,
1892 				    optarg));
1893 				if (ld_sym_add_u(optarg, ofl,
1894 				    MSG_STR_COMMAND) == (Sym_desc *)S_ERROR)
1895 					return (S_ERROR);
1896 				break;
1897 			case 'z':
1898 				DBG_CALL(Dbg_args_option(ofl->ofl_lml, ndx, c,
1899 				    optarg));
1900 				if ((strncmp(optarg, MSG_ORIG(MSG_ARG_LD32),
1901 				    MSG_ARG_LD32_SIZE) == 0) ||
1902 				    (strncmp(optarg, MSG_ORIG(MSG_ARG_LD64),
1903 				    MSG_ARG_LD64_SIZE) == 0)) {
1904 					if (createargv(ofl, 0) == S_ERROR)
1905 						return (S_ERROR);
1906 				} else if (strcmp(optarg,
1907 				    MSG_ORIG(MSG_ARG_ALLEXTRT)) == 0) {
1908 					ofl->ofl_flags1 |= FLG_OF1_ALLEXRT;
1909 					ofl->ofl_flags1 &= ~FLG_OF1_WEAKEXT;
1910 				} else if (strcmp(optarg,
1911 				    MSG_ORIG(MSG_ARG_WEAKEXT)) == 0) {
1912 					ofl->ofl_flags1 |= FLG_OF1_WEAKEXT;
1913 					ofl->ofl_flags1 &= ~FLG_OF1_ALLEXRT;
1914 				} else if (strcmp(optarg,
1915 				    MSG_ORIG(MSG_ARG_DFLEXTRT)) == 0) {
1916 					ofl->ofl_flags1 &=
1917 					    ~(FLG_OF1_ALLEXRT |
1918 					    FLG_OF1_WEAKEXT);
1919 				} else if (strcmp(optarg,
1920 				    MSG_ORIG(MSG_ARG_DIRECT)) == 0) {
1921 					ofl->ofl_flags1 |= FLG_OF1_ZDIRECT;
1922 					ofl->ofl_guideflags |= FLG_OFG_NO_DB;
1923 				} else if (strcmp(optarg,
1924 				    MSG_ORIG(MSG_ARG_NODIRECT)) == 0) {
1925 					ofl->ofl_flags1 &= ~FLG_OF1_ZDIRECT;
1926 					ofl->ofl_guideflags |= FLG_OFG_NO_DB;
1927 				} else if (strcmp(optarg,
1928 				    MSG_ORIG(MSG_ARG_IGNORE)) == 0) {
1929 					ofl->ofl_flags1 |= FLG_OF1_IGNORE;
1930 				} else if (strcmp(optarg,
1931 				    MSG_ORIG(MSG_ARG_RECORD)) == 0) {
1932 					ofl->ofl_flags1 &= ~FLG_OF1_IGNORE;
1933 				} else if (strcmp(optarg,
1934 				    MSG_ORIG(MSG_ARG_LAZYLOAD)) == 0) {
1935 					ofl->ofl_flags1 |= FLG_OF1_LAZYLD;
1936 					ofl->ofl_guideflags |= FLG_OFG_NO_LAZY;
1937 				} else if (strcmp(optarg,
1938 				    MSG_ORIG(MSG_ARG_NOLAZYLOAD)) == 0) {
1939 					ofl->ofl_flags1 &= ~ FLG_OF1_LAZYLD;
1940 					ofl->ofl_guideflags |= FLG_OFG_NO_LAZY;
1941 				} else if (strcmp(optarg,
1942 				    MSG_ORIG(MSG_ARG_GROUPPERM)) == 0) {
1943 					ofl->ofl_flags1 |= FLG_OF1_GRPPRM;
1944 				} else if (strcmp(optarg,
1945 				    MSG_ORIG(MSG_ARG_NOGROUPPERM)) == 0) {
1946 					ofl->ofl_flags1 &= ~FLG_OF1_GRPPRM;
1947 				} else if (strncmp(optarg,
1948 				    MSG_ORIG(MSG_ARG_INITARRAY),
1949 				    MSG_ARG_INITARRAY_SIZE) == 0) {
1950 					if (((sdp = ld_sym_add_u(optarg +
1951 					    MSG_ARG_INITARRAY_SIZE, ofl,
1952 					    MSG_STR_COMMAND)) ==
1953 					    (Sym_desc *)S_ERROR) ||
1954 					    (aplist_append(&ofl->ofl_initarray,
1955 					    sdp, AL_CNT_OFL_ARRAYS) == NULL))
1956 						return (S_ERROR);
1957 				} else if (strncmp(optarg,
1958 				    MSG_ORIG(MSG_ARG_FINIARRAY),
1959 				    MSG_ARG_FINIARRAY_SIZE) == 0) {
1960 					if (((sdp = ld_sym_add_u(optarg +
1961 					    MSG_ARG_FINIARRAY_SIZE, ofl,
1962 					    MSG_STR_COMMAND)) ==
1963 					    (Sym_desc *)S_ERROR) ||
1964 					    (aplist_append(&ofl->ofl_finiarray,
1965 					    sdp, AL_CNT_OFL_ARRAYS) == NULL))
1966 						return (S_ERROR);
1967 				} else if (strncmp(optarg,
1968 				    MSG_ORIG(MSG_ARG_PREINITARRAY),
1969 				    MSG_ARG_PREINITARRAY_SIZE) == 0) {
1970 					if (((sdp = ld_sym_add_u(optarg +
1971 					    MSG_ARG_PREINITARRAY_SIZE, ofl,
1972 					    MSG_STR_COMMAND)) ==
1973 					    (Sym_desc *)S_ERROR) ||
1974 					    (aplist_append(&ofl->ofl_preiarray,
1975 					    sdp, AL_CNT_OFL_ARRAYS) == NULL))
1976 						return (S_ERROR);
1977 				} else if (strncmp(optarg,
1978 				    MSG_ORIG(MSG_ARG_RTLDINFO),
1979 				    MSG_ARG_RTLDINFO_SIZE) == 0) {
1980 					if (((sdp = ld_sym_add_u(optarg +
1981 					    MSG_ARG_RTLDINFO_SIZE, ofl,
1982 					    MSG_STR_COMMAND)) ==
1983 					    (Sym_desc *)S_ERROR) ||
1984 					    (aplist_append(&ofl->ofl_rtldinfo,
1985 					    sdp, AL_CNT_OFL_ARRAYS) == NULL))
1986 						return (S_ERROR);
1987 				} else if (strncmp(optarg,
1988 				    MSG_ORIG(MSG_ARG_DTRACE),
1989 				    MSG_ARG_DTRACE_SIZE) == 0) {
1990 					if ((sdp = ld_sym_add_u(optarg +
1991 					    MSG_ARG_DTRACE_SIZE, ofl,
1992 					    MSG_STR_COMMAND)) ==
1993 					    (Sym_desc *)S_ERROR)
1994 						return (S_ERROR);
1995 					ofl->ofl_dtracesym = sdp;
1996 				} else if (strcmp(optarg,
1997 				    MSG_ORIG(MSG_ARG_RESCAN_NOW)) == 0) {
1998 					if (ld_rescan_archives(ofl, 0, ndx) ==
1999 					    S_ERROR)
2000 						return (S_ERROR);
2001 				} else if (strcmp(optarg,
2002 				    MSG_ORIG(MSG_ARG_RESCAN_START)) == 0) {
2003 					ofl->ofl_ars_gsndx = ofl->ofl_arscnt;
2004 					ofl->ofl_ars_gsandx = ndx;
2005 				} else if (strcmp(optarg,
2006 				    MSG_ORIG(MSG_ARG_RESCAN_END)) == 0) {
2007 					if (ld_rescan_archives(ofl, 1, ndx) ==
2008 					    S_ERROR)
2009 						return (S_ERROR);
2010 				} else if (strcmp(optarg,
2011 				    MSG_ORIG(MSG_ARG_DEFERRED)) == 0) {
2012 					ofl->ofl_flags1 |= FLG_OF1_DEFERRED;
2013 				} else if (strcmp(optarg,
2014 				    MSG_ORIG(MSG_ARG_NODEFERRED)) == 0) {
2015 					ofl->ofl_flags1 &= ~FLG_OF1_DEFERRED;
2016 				}
2017 			default:
2018 				break;
2019 		}
2020 
2021 		/*
2022 		 * Update the argument index for the next getopt() iteration.
2023 		 */
2024 		ndx = optind;
2025 	}
2026 	return (1);
2027 }
2028 
2029 /*
2030  *
2031  * Pass 1 -- process_flags: collects all options and sets flags
2032  */
2033 static uintptr_t
2034 process_flags_com(Ofl_desc *ofl, int argc, char **argv, int *usage)
2035 {
2036 	for (; optind < argc; optind++) {
2037 		/*
2038 		 * If we detect some more options return to getopt().
2039 		 * Checking argv[optind][1] against null prevents a forever
2040 		 * loop if an unadorned `-' argument is passed to us.
2041 		 */
2042 		while ((optind < argc) && (argv[optind][0] == '-')) {
2043 			if (argv[optind][1] != '\0') {
2044 				if (parseopt_pass1(ofl, argc, argv,
2045 				    usage) == S_ERROR)
2046 					return (S_ERROR);
2047 			} else if (++optind < argc)
2048 				continue;
2049 		}
2050 		if (optind >= argc)
2051 			break;
2052 		ofl->ofl_objscnt++;
2053 	}
2054 
2055 	/* Did an unterminated archive group run off the end? */
2056 	if (ofl->ofl_ars_gsandx > 0) {
2057 		ld_eprintf(ofl, ERR_FATAL, MSG_INTL(MSG_ARG_AR_GRP_BAD),
2058 		    MSG_INTL(MSG_MARG_AR_GRP_START),
2059 		    MSG_INTL(MSG_MARG_AR_GRP_END));
2060 		return (S_ERROR);
2061 	}
2062 
2063 	return (1);
2064 }
2065 
2066 uintptr_t
2067 ld_process_flags(Ofl_desc *ofl, int argc, char **argv)
2068 {
2069 	int	usage = 0;	/* Collect all argument errors before exit */
2070 
2071 	if (argc < 2) {
2072 		usage_mesg(FALSE);
2073 		return (S_ERROR);
2074 	}
2075 
2076 	/*
2077 	 * Option handling
2078 	 */
2079 	opterr = 0;
2080 	optind = 1;
2081 	if (process_flags_com(ofl, argc, argv, &usage) == S_ERROR)
2082 		return (S_ERROR);
2083 
2084 	/*
2085 	 * Having parsed everything, did we have any usage errors.
2086 	 */
2087 	if (usage) {
2088 		eprintf(ofl->ofl_lml, ERR_FATAL, MSG_INTL(MSG_ARG_USEHELP));
2089 		return (S_ERROR);
2090 	}
2091 
2092 	return (check_flags(ofl, argc));
2093 }
2094 
2095 /*
2096  * Pass 2 -- process_files: skips the flags collected in pass 1 and processes
2097  * files.
2098  */
2099 static uintptr_t
2100 process_files_com(Ofl_desc *ofl, int argc, char **argv)
2101 {
2102 	for (; optind < argc; optind++) {
2103 		int		fd;
2104 		uintptr_t	open_ret;
2105 		char		*path;
2106 		Rej_desc	rej = { 0 };
2107 
2108 		/*
2109 		 * If we detect some more options return to getopt().
2110 		 * Checking argv[optind][1] against null prevents a forever
2111 		 * loop if an unadorned `-' argument is passed to us.
2112 		 */
2113 		while ((optind < argc) && (argv[optind][0] == '-')) {
2114 			if (argv[optind][1] != '\0') {
2115 				if (parseopt_pass2(ofl, argc, argv) == S_ERROR)
2116 					return (S_ERROR);
2117 			} else if (++optind < argc)
2118 				continue;
2119 		}
2120 		if (optind >= argc)
2121 			break;
2122 
2123 		path = argv[optind];
2124 		if ((fd = open(path, O_RDONLY)) == -1) {
2125 			int err = errno;
2126 
2127 			ld_eprintf(ofl, ERR_FATAL,
2128 			    MSG_INTL(MSG_SYS_OPEN), path, strerror(err));
2129 			continue;
2130 		}
2131 
2132 		DBG_CALL(Dbg_args_file(ofl->ofl_lml, optind, path));
2133 
2134 		open_ret = ld_process_open(path, path, &fd, ofl,
2135 		    (FLG_IF_CMDLINE | FLG_IF_NEEDED), &rej, NULL);
2136 		if (fd != -1)
2137 			(void) close(fd);
2138 		if (open_ret == S_ERROR)
2139 			return (S_ERROR);
2140 
2141 		/*
2142 		 * Check for mismatched input.
2143 		 */
2144 		if (rej.rej_type) {
2145 			Conv_reject_desc_buf_t rej_buf;
2146 
2147 			ld_eprintf(ofl, ERR_FATAL,
2148 			    MSG_INTL(reject[rej.rej_type]),
2149 			    rej.rej_name ? rej.rej_name :
2150 			    MSG_INTL(MSG_STR_UNKNOWN),
2151 			    conv_reject_desc(&rej, &rej_buf,
2152 			    ld_targ.t_m.m_mach));
2153 			return (1);
2154 		}
2155 	}
2156 	return (1);
2157 }
2158 
2159 uintptr_t
2160 ld_process_files(Ofl_desc *ofl, int argc, char **argv)
2161 {
2162 	DBG_CALL(Dbg_basic_files(ofl->ofl_lml));
2163 
2164 	/*
2165 	 * Process command line files (taking into account any applicable
2166 	 * preceding flags).  Return if any fatal errors have occurred.
2167 	 */
2168 	opterr = 0;
2169 	optind = 1;
2170 	if (process_files_com(ofl, argc, argv) == S_ERROR)
2171 		return (S_ERROR);
2172 	if (ofl->ofl_flags & FLG_OF_FATAL)
2173 		return (1);
2174 
2175 	/*
2176 	 * Guidance: Use -B direct/nodirect or -z direct/nodirect.
2177 	 *
2178 	 * This is a backstop for the case where the link had no dependencies.
2179 	 * Otherwise, it will get caught by ld_process_ifl(). We need both,
2180 	 * because -z direct is positional, and its value at the time where
2181 	 * the first dependency is seen might be different than it is now.
2182 	 */
2183 	if ((ofl->ofl_flags & FLG_OF_DYNAMIC) &&
2184 	    OFL_GUIDANCE(ofl, FLG_OFG_NO_DB)) {
2185 		ld_eprintf(ofl, ERR_GUIDANCE, MSG_INTL(MSG_GUIDE_DIRECT));
2186 		ofl->ofl_guideflags |= FLG_OFG_NO_DB;
2187 	}
2188 
2189 	/*
2190 	 * Now that all command line files have been processed see if there are
2191 	 * any additional `needed' shared object dependencies.
2192 	 */
2193 	if (ofl->ofl_soneed)
2194 		if (ld_finish_libs(ofl) == S_ERROR)
2195 			return (S_ERROR);
2196 
2197 	/*
2198 	 * If rescanning archives is enabled, do so now to determine whether
2199 	 * there might still be members extracted to satisfy references from any
2200 	 * explicit objects.  Continue until no new objects are extracted.  Note
2201 	 * that this pass is carried out *after* processing any implicit objects
2202 	 * (above) as they may already have resolved any undefined references
2203 	 * from any explicit dependencies.
2204 	 */
2205 	if (ofl->ofl_flags1 & FLG_OF1_RESCAN) {
2206 		if (ld_rescan_archives(ofl, 0, argc) == S_ERROR)
2207 			return (S_ERROR);
2208 		if (ofl->ofl_flags & FLG_OF_FATAL)
2209 			return (1);
2210 	}
2211 
2212 	/*
2213 	 * If debugging, provide statistics on each archives extraction, or flag
2214 	 * any archive that has provided no members.  Note that this could be a
2215 	 * nice place to free up much of the archive infrastructure, as we've
2216 	 * extracted any members we need.  However, as we presently don't free
2217 	 * anything under ld(1) there's not much point in proceeding further.
2218 	 */
2219 	DBG_CALL(Dbg_statistics_ar(ofl));
2220 
2221 	/*
2222 	 * If any version definitions have been established, either via input
2223 	 * from a mapfile or from the input relocatable objects, make sure any
2224 	 * version dependencies are satisfied, and version symbols created.
2225 	 */
2226 	if (ofl->ofl_verdesc)
2227 		if (ld_vers_check_defs(ofl) == S_ERROR)
2228 			return (S_ERROR);
2229 
2230 	/*
2231 	 * If input section ordering was specified within some segment
2232 	 * using a mapfile, verify that the expected sections were seen.
2233 	 */
2234 	if (ofl->ofl_flags & FLG_OF_IS_ORDER)
2235 		ld_ent_check(ofl);
2236 
2237 	return (1);
2238 }
2239 
2240 uintptr_t
2241 ld_init_strings(Ofl_desc *ofl)
2242 {
2243 	uint_t	stflags;
2244 
2245 	if (ofl->ofl_flags1 & FLG_OF1_NCSTTAB)
2246 		stflags = 0;
2247 	else
2248 		stflags = FLG_STNEW_COMPRESS;
2249 
2250 	if (((ofl->ofl_shdrsttab = st_new(stflags)) == NULL) ||
2251 	    ((ofl->ofl_strtab = st_new(stflags)) == NULL) ||
2252 	    ((ofl->ofl_dynstrtab = st_new(stflags)) == NULL))
2253 		return (S_ERROR);
2254 
2255 	return (0);
2256 }
2257