opcodes/
[binutils-gdb.git] / bfd / vms-lib.c
blobfa23b788e2509f6caa7303ccd34bba80fdfe436a
1 /* BFD back-end for VMS archive files.
3 Copyright 2010, 2011 Free Software Foundation, Inc.
4 Written by Tristan Gingold <gingold@adacore.com>, AdaCore.
6 This file is part of BFD, the Binary File Descriptor library.
8 This program is free software; you can redistribute it and/or modify
9 it under the terms of the GNU General Public License as published by
10 the Free Software Foundation; either version 3 of the License, or
11 (at your option) any later version.
13 This program is distributed in the hope that it will be useful,
14 but WITHOUT ANY WARRANTY; without even the implied warranty of
15 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
16 GNU General Public License for more details.
18 You should have received a copy of the GNU General Public License
19 along with this program; if not, write to the Free Software
20 Foundation, Inc., 51 Franklin Street - Fifth Floor, Boston,
21 MA 02110-1301, USA. */
23 #include "sysdep.h"
24 #include "bfd.h"
25 #include "libbfd.h"
26 #include "safe-ctype.h"
27 #include "bfdver.h"
28 #include "vms.h"
29 #include "vms/lbr.h"
30 #include "vms/dcx.h"
32 /* The standard VMS disk block size. */
33 #ifndef VMS_BLOCK_SIZE
34 #define VMS_BLOCK_SIZE 512
35 #endif
37 /* Maximum key length (which is also the maximum symbol length in archive). */
38 #define MAX_KEYLEN 128
39 #define MAX_EKEYLEN 1024
41 /* DCX Submaps. */
43 struct dcxsbm_desc
45 unsigned char min_char;
46 unsigned char max_char;
47 unsigned char *flags;
48 unsigned char *nodes;
49 unsigned short *next;
52 /* Kind of library. Used to filter in archive_p. */
54 enum vms_lib_kind
56 vms_lib_vax,
57 vms_lib_alpha,
58 vms_lib_ia64,
59 vms_lib_txt
62 /* Back-end private data. */
64 struct lib_tdata
66 /* Standard tdata for an archive. But we don't use many fields. */
67 struct artdata artdata;
69 /* Major version. */
70 unsigned char ver;
72 /* Type of the archive. */
73 unsigned char type;
75 /* Kind of archive. Summary of its type. */
76 enum vms_lib_kind kind;
78 /* Total size of the mhd (element header). */
79 unsigned int mhd_size;
81 /* Creation date. */
82 unsigned int credat_lo;
83 unsigned int credat_hi;
85 /* Vector of modules (archive elements), already sorted. */
86 unsigned int nbr_modules;
87 struct carsym *modules;
88 bfd **cache;
90 /* DCX (decompression) data. */
91 unsigned int nbr_dcxsbm;
92 struct dcxsbm_desc *dcxsbm;
95 #define bfd_libdata(bfd) ((struct lib_tdata *)((bfd)->tdata.any))
97 /* End-Of-Text pattern. This is a special record to mark the end of file. */
99 static const unsigned char eotdesc[] = { 0x03, 0x00, 0x77, 0x00, 0x77, 0x00 };
101 /* Describe the current state of carsym entries while building the archive
102 table of content. Things are simple with Alpha archives as the number
103 of entries is known, but with IA64 archives a entry can make a reference
104 to severals members. Therefore we must be able to extend the table on the
105 fly, but it should be allocated on the bfd - which doesn't support realloc.
106 To reduce the overhead, the table is initially allocated in the BFD's
107 objalloc and extended if necessary on the heap. In the later case, it
108 is finally copied to the BFD's objalloc so that it will automatically be
109 freed. */
111 struct carsym_mem
113 /* The table of content. */
114 struct carsym *idx;
116 /* Number of entries used in the table. */
117 unsigned int nbr;
119 /* Maximum number of entries. */
120 unsigned int max;
122 /* If true, the table was reallocated on the heap. If false, it is still
123 in the BFD's objalloc. */
124 bfd_boolean realloced;
127 /* Simply add a name to the index. */
129 static bfd_boolean
130 vms_add_index (struct carsym_mem *cs, char *name,
131 unsigned int idx_vbn, unsigned int idx_off)
133 if (cs->nbr == cs->max)
135 struct carsym *n;
137 cs->max = 2 * cs->max + 32;
139 if (!cs->realloced)
141 n = bfd_malloc2 (cs->max, sizeof (struct carsym));
142 if (n == NULL)
143 return FALSE;
144 memcpy (n, cs->idx, cs->nbr * sizeof (struct carsym));
145 /* And unfortunately we can't free cs->idx. */
147 else
149 n = bfd_realloc_or_free (cs->idx, cs->nbr * sizeof (struct carsym));
150 if (n == NULL)
151 return FALSE;
153 cs->idx = n;
154 cs->realloced = TRUE;
156 cs->idx[cs->nbr].file_offset = (idx_vbn - 1) * VMS_BLOCK_SIZE + idx_off;
157 cs->idx[cs->nbr].name = name;
158 cs->nbr++;
159 return TRUE;
162 /* Follow all member of a lns list (pointed by RFA) and add indexes for
163 NAME. Return FALSE in case of error. */
165 static bfd_boolean
166 vms_add_indexes_from_list (bfd *abfd, struct carsym_mem *cs, char *name,
167 struct vms_rfa *rfa)
169 struct vms_lns lns;
170 unsigned int vbn;
171 file_ptr off;
173 while (1)
175 vbn = bfd_getl32 (rfa->vbn);
176 if (vbn == 0)
177 return TRUE;
179 /* Read the LHS. */
180 off = (vbn - 1) * VMS_BLOCK_SIZE + bfd_getl16 (rfa->offset);
181 if (bfd_seek (abfd, off, SEEK_SET) != 0
182 || bfd_bread (&lns, sizeof (lns), abfd) != sizeof (lns))
183 return FALSE;
185 if (!vms_add_index (cs, name,
186 bfd_getl32 (lns.modrfa.vbn),
187 bfd_getl16 (lns.modrfa.offset)))
188 return FALSE;
190 rfa = &lns.nxtrfa;
194 /* Read block VBN from ABFD and store it into BLK. Return FALSE in case of error. */
196 static bfd_boolean
197 vms_read_block (bfd *abfd, unsigned int vbn, void *blk)
199 file_ptr off;
201 off = (vbn - 1) * VMS_BLOCK_SIZE;
202 if (bfd_seek (abfd, off, SEEK_SET) != 0
203 || bfd_bread (blk, VMS_BLOCK_SIZE, abfd) != VMS_BLOCK_SIZE)
204 return FALSE;
206 return TRUE;
209 /* Write the content of BLK to block VBN of ABFD. Return FALSE in case of error. */
211 static bfd_boolean
212 vms_write_block (bfd *abfd, unsigned int vbn, void *blk)
214 file_ptr off;
216 off = (vbn - 1) * VMS_BLOCK_SIZE;
217 if (bfd_seek (abfd, off, SEEK_SET) != 0
218 || bfd_bwrite (blk, VMS_BLOCK_SIZE, abfd) != VMS_BLOCK_SIZE)
219 return FALSE;
221 return TRUE;
224 /* Read index block VBN and put the entry in **IDX (which is updated).
225 If the entry is indirect, recurse. */
227 static bfd_boolean
228 vms_traverse_index (bfd *abfd, unsigned int vbn, struct carsym_mem *cs)
230 struct vms_indexdef indexdef;
231 file_ptr off;
232 unsigned char *p;
233 unsigned char *endp;
235 /* Read the index block. */
236 BFD_ASSERT (sizeof (indexdef) == VMS_BLOCK_SIZE);
237 if (!vms_read_block (abfd, vbn, &indexdef))
238 return FALSE;
240 /* Traverse it. */
241 p = &indexdef.keys[0];
242 endp = p + bfd_getl16 (indexdef.used);
243 while (p < endp)
245 unsigned int idx_vbn;
246 unsigned int idx_off;
247 unsigned int keylen;
248 unsigned char *keyname;
249 unsigned int flags;
251 /* Extract key length. */
252 if (bfd_libdata (abfd)->ver == LBR_MAJORID)
254 struct vms_idx *ridx = (struct vms_idx *)p;
256 idx_vbn = bfd_getl32 (ridx->rfa.vbn);
257 idx_off = bfd_getl16 (ridx->rfa.offset);
259 keylen = ridx->keylen;
260 flags = 0;
261 keyname = ridx->keyname;
263 else if (bfd_libdata (abfd)->ver == LBR_ELFMAJORID)
265 struct vms_elfidx *ridx = (struct vms_elfidx *)p;
267 idx_vbn = bfd_getl32 (ridx->rfa.vbn);
268 idx_off = bfd_getl16 (ridx->rfa.offset);
270 keylen = bfd_getl16 (ridx->keylen);
271 flags = ridx->flags;
272 keyname = ridx->keyname;
274 else
275 return FALSE;
277 /* Illegal value. */
278 if (idx_vbn == 0)
279 return FALSE;
281 /* Point to the next index entry. */
282 p = keyname + keylen;
284 if (idx_off == RFADEF__C_INDEX)
286 /* Indirect entry. Recurse. */
287 if (!vms_traverse_index (abfd, idx_vbn, cs))
288 return FALSE;
290 else
292 /* Add a new entry. */
293 char *name;
295 if (flags & ELFIDX__SYMESC)
297 /* Extended key name. */
298 unsigned int noff = 0;
299 unsigned int koff;
300 unsigned int kvbn;
301 struct vms_kbn *kbn;
302 unsigned char kblk[VMS_BLOCK_SIZE];
304 /* Sanity check. */
305 if (keylen != sizeof (struct vms_kbn))
306 return FALSE;
308 kbn = (struct vms_kbn *)keyname;
309 keylen = bfd_getl16 (kbn->keylen);
311 name = bfd_alloc (abfd, keylen + 1);
312 if (name == NULL)
313 return FALSE;
314 kvbn = bfd_getl32 (kbn->rfa.vbn);
315 koff = bfd_getl16 (kbn->rfa.offset);
317 /* Read the key, chunk by chunk. */
320 unsigned int klen;
322 if (!vms_read_block (abfd, kvbn, kblk))
323 return FALSE;
324 kbn = (struct vms_kbn *)(kblk + koff);
325 klen = bfd_getl16 (kbn->keylen);
326 kvbn = bfd_getl32 (kbn->rfa.vbn);
327 koff = bfd_getl16 (kbn->rfa.offset);
329 memcpy (name + noff, kbn + 1, klen);
330 noff += klen;
332 while (kvbn != 0);
334 /* Sanity check. */
335 if (noff != keylen)
336 return FALSE;
338 else
340 /* Usual key name. */
341 name = bfd_alloc (abfd, keylen + 1);
342 if (name == NULL)
343 return FALSE;
345 memcpy (name, keyname, keylen);
347 name[keylen] = 0;
349 if (flags & ELFIDX__LISTRFA)
351 struct vms_lhs lhs;
353 /* Read the LHS. */
354 off = (idx_vbn - 1) * VMS_BLOCK_SIZE + idx_off;
355 if (bfd_seek (abfd, off, SEEK_SET) != 0
356 || bfd_bread (&lhs, sizeof (lhs), abfd) != sizeof (lhs))
357 return FALSE;
359 /* FIXME: this adds extra entries that were not accounted. */
360 if (!vms_add_indexes_from_list (abfd, cs, name, &lhs.ng_g_rfa))
361 return FALSE;
362 if (!vms_add_indexes_from_list (abfd, cs, name, &lhs.ng_wk_rfa))
363 return FALSE;
364 if (!vms_add_indexes_from_list (abfd, cs, name, &lhs.g_g_rfa))
365 return FALSE;
366 if (!vms_add_indexes_from_list (abfd, cs, name, &lhs.g_wk_rfa))
367 return FALSE;
369 else
371 if (!vms_add_index (cs, name, idx_vbn, idx_off))
372 return FALSE;
377 return TRUE;
380 /* Read index #IDX, which must have NBREL entries. */
382 static struct carsym *
383 vms_lib_read_index (bfd *abfd, int idx, unsigned int *nbrel)
385 struct vms_idd idd;
386 unsigned int flags;
387 unsigned int vbn;
388 struct carsym *csbuf;
389 struct carsym_mem csm;
391 /* Read index desription. */
392 if (bfd_seek (abfd, LHD_IDXDESC + idx * IDD_LENGTH, SEEK_SET) != 0
393 || bfd_bread (&idd, sizeof (idd), abfd) != sizeof (idd))
394 return NULL;
396 /* Sanity checks. */
397 flags = bfd_getl16 (idd.flags);
398 if (!(flags & IDD__FLAGS_ASCII)
399 || !(flags & IDD__FLAGS_VARLENIDX))
400 return NULL;
402 csbuf = bfd_alloc (abfd, *nbrel * sizeof (struct carsym));
403 if (csbuf == NULL)
404 return NULL;
406 csm.max = *nbrel;
407 csm.nbr = 0;
408 csm.realloced = FALSE;
409 csm.idx = csbuf;
411 /* Note: if the index is empty, there is no block to traverse. */
412 vbn = bfd_getl32 (idd.vbn);
413 if (vbn != 0 && !vms_traverse_index (abfd, vbn, &csm))
415 if (csm.realloced && csm.idx != NULL)
416 free (csm.idx);
418 /* Note: in case of error, we can free what was allocated on the
419 BFD's objalloc. */
420 bfd_release (abfd, csbuf);
421 return NULL;
424 if (csm.realloced)
426 /* There are more entries than the first estimate. Allocate on
427 the BFD's objalloc. */
428 csbuf = bfd_alloc (abfd, csm.nbr * sizeof (struct carsym));
429 if (csbuf == NULL)
430 return NULL;
431 memcpy (csbuf, csm.idx, csm.nbr * sizeof (struct carsym));
432 free (csm.idx);
433 *nbrel = csm.nbr;
435 return csbuf;
438 /* Standard function. */
440 static const bfd_target *
441 _bfd_vms_lib_archive_p (bfd *abfd, enum vms_lib_kind kind)
443 struct vms_lhd lhd;
444 unsigned int sanity;
445 unsigned int majorid;
446 struct lib_tdata *tdata_hold;
447 struct lib_tdata *tdata;
448 unsigned int dcxvbn;
449 unsigned int nbr_ent;
451 /* Read header. */
452 if (bfd_bread (&lhd, sizeof (lhd), abfd) != sizeof (lhd))
454 if (bfd_get_error () != bfd_error_system_call)
455 bfd_set_error (bfd_error_wrong_format);
456 return NULL;
459 /* Check sanity (= magic) number. */
460 sanity = bfd_getl32 (lhd.sanity);
461 if (!(sanity == LHD_SANEID3
462 || sanity == LHD_SANEID6
463 || sanity == LHD_SANEID_DCX))
465 bfd_set_error (bfd_error_wrong_format);
466 return NULL;
468 majorid = bfd_getl32 (lhd.majorid);
470 /* Check archive kind. */
471 switch (kind)
473 case vms_lib_alpha:
474 if ((lhd.type != LBR__C_TYP_EOBJ && lhd.type != LBR__C_TYP_ESHSTB)
475 || majorid != LBR_MAJORID
476 || lhd.nindex != 2)
478 bfd_set_error (bfd_error_wrong_format);
479 return NULL;
481 break;
482 case vms_lib_ia64:
483 if ((lhd.type != LBR__C_TYP_IOBJ && lhd.type != LBR__C_TYP_ISHSTB)
484 || majorid != LBR_ELFMAJORID
485 || lhd.nindex != 2)
487 bfd_set_error (bfd_error_wrong_format);
488 return NULL;
490 break;
491 case vms_lib_txt:
492 if ((lhd.type != LBR__C_TYP_TXT
493 && lhd.type != LBR__C_TYP_MLB
494 && lhd.type != LBR__C_TYP_HLP)
495 || majorid != LBR_MAJORID
496 || lhd.nindex != 1)
498 bfd_set_error (bfd_error_wrong_format);
499 return NULL;
501 break;
502 default:
503 abort ();
506 /* Allocate and initialize private data. */
507 tdata_hold = bfd_libdata (abfd);
508 tdata = (struct lib_tdata *) bfd_zalloc (abfd, sizeof (struct lib_tdata));
509 if (tdata == NULL)
510 return NULL;
511 abfd->tdata.any = (void *)tdata;
512 tdata->ver = majorid;
513 tdata->mhd_size = MHD__C_USRDAT + lhd.mhdusz;
514 tdata->type = lhd.type;
515 tdata->kind = kind;
516 tdata->credat_lo = bfd_getl32 (lhd.credat + 0);
517 tdata->credat_hi = bfd_getl32 (lhd.credat + 4);
519 /* Read indexes. */
520 tdata->nbr_modules = bfd_getl32 (lhd.modcnt);
521 tdata->artdata.symdef_count = bfd_getl32 (lhd.idxcnt) - tdata->nbr_modules;
522 nbr_ent = tdata->nbr_modules;
523 tdata->modules = vms_lib_read_index (abfd, 0, &nbr_ent);
524 if (tdata->modules == NULL || nbr_ent != tdata->nbr_modules)
525 goto err;
526 if (lhd.nindex == 2)
528 nbr_ent = tdata->artdata.symdef_count;
529 tdata->artdata.symdefs = vms_lib_read_index (abfd, 1, &nbr_ent);
530 if (tdata->artdata.symdefs == NULL)
531 goto err;
532 /* Only IA64 archives may have more entries in the index that what
533 was declared. */
534 if (nbr_ent != tdata->artdata.symdef_count
535 && kind != vms_lib_ia64)
536 goto err;
537 tdata->artdata.symdef_count = nbr_ent;
539 tdata->cache = bfd_zalloc (abfd, sizeof (bfd *) * tdata->nbr_modules);
540 if (tdata->cache == NULL)
541 goto err;
543 /* Read DCX submaps. */
544 dcxvbn = bfd_getl32 (lhd.dcxmapvbn);
545 if (dcxvbn != 0)
547 unsigned char buf_reclen[4];
548 unsigned int reclen;
549 unsigned char *buf;
550 struct vms_dcxmap *map;
551 unsigned int sbm_off;
552 unsigned int i;
554 if (bfd_seek (abfd, (dcxvbn - 1) * VMS_BLOCK_SIZE, SEEK_SET) != 0
555 || bfd_bread (buf_reclen, sizeof (buf_reclen), abfd)
556 != sizeof (buf_reclen))
557 goto err;
558 reclen = bfd_getl32 (buf_reclen);
559 buf = bfd_malloc (reclen);
560 if (buf == NULL)
561 goto err;
562 if (bfd_bread (buf, reclen, abfd) != reclen)
564 free (buf);
565 goto err;
567 map = (struct vms_dcxmap *)buf;
568 tdata->nbr_dcxsbm = bfd_getl16 (map->nsubs);
569 sbm_off = bfd_getl16 (map->sub0);
570 tdata->dcxsbm = (struct dcxsbm_desc *)bfd_alloc
571 (abfd, tdata->nbr_dcxsbm * sizeof (struct dcxsbm_desc));
572 for (i = 0; i < tdata->nbr_dcxsbm; i++)
574 struct vms_dcxsbm *sbm = (struct vms_dcxsbm *) (buf + sbm_off);
575 struct dcxsbm_desc *sbmdesc = &tdata->dcxsbm[i];
576 unsigned int sbm_len;
577 unsigned int sbm_sz;
578 unsigned int off;
579 unsigned char *data = (unsigned char *)sbm;
580 unsigned char *buf1;
581 unsigned int l, j;
583 sbm_sz = bfd_getl16 (sbm->size);
584 sbm_off += sbm_sz;
585 BFD_ASSERT (sbm_off <= reclen);
587 sbmdesc->min_char = sbm->min_char;
588 BFD_ASSERT (sbmdesc->min_char == 0);
589 sbmdesc->max_char = sbm->max_char;
590 sbm_len = sbmdesc->max_char - sbmdesc->min_char + 1;
591 l = (2 * sbm_len + 7) / 8;
592 BFD_ASSERT
593 (sbm_sz >= sizeof (struct vms_dcxsbm) + l + 3 * sbm_len
594 || (tdata->nbr_dcxsbm == 1
595 && sbm_sz >= sizeof (struct vms_dcxsbm) + l + sbm_len));
596 sbmdesc->flags = (unsigned char *)bfd_alloc (abfd, l);
597 memcpy (sbmdesc->flags, data + bfd_getl16 (sbm->flags), l);
598 sbmdesc->nodes = (unsigned char *)bfd_alloc (abfd, 2 * sbm_len);
599 memcpy (sbmdesc->nodes, data + bfd_getl16 (sbm->nodes), 2 * sbm_len);
600 off = bfd_getl16 (sbm->next);
601 if (off != 0)
603 /* Read the 'next' array. */
604 sbmdesc->next = (unsigned short *)bfd_alloc
605 (abfd, sbm_len * sizeof (unsigned short));
606 buf1 = data + off;
607 for (j = 0; j < sbm_len; j++)
608 sbmdesc->next[j] = bfd_getl16 (buf1 + j * 2);
610 else
612 /* There is no next array if there is only one submap. */
613 BFD_ASSERT (tdata->nbr_dcxsbm == 1);
614 sbmdesc->next = NULL;
617 free (buf);
619 else
621 tdata->nbr_dcxsbm = 0;
624 /* The map is always present. Also mark shared image library. */
625 abfd->has_armap = TRUE;
626 if (tdata->type == LBR__C_TYP_ESHSTB || tdata->type == LBR__C_TYP_ISHSTB)
627 abfd->is_thin_archive = TRUE;
629 return abfd->xvec;
631 err:
632 bfd_release (abfd, tdata);
633 abfd->tdata.any = (void *)tdata_hold;;
634 return NULL;
637 /* Standard function for alpha libraries. */
639 const bfd_target *
640 _bfd_vms_lib_alpha_archive_p (bfd *abfd)
642 return _bfd_vms_lib_archive_p (abfd, vms_lib_alpha);
645 /* Standard function for ia64 libraries. */
647 const bfd_target *
648 _bfd_vms_lib_ia64_archive_p (bfd *abfd)
650 return _bfd_vms_lib_archive_p (abfd, vms_lib_ia64);
653 /* Standard function for text libraries. */
655 static const bfd_target *
656 _bfd_vms_lib_txt_archive_p (bfd *abfd)
658 return _bfd_vms_lib_archive_p (abfd, vms_lib_txt);
661 /* Standard bfd function. */
663 static bfd_boolean
664 _bfd_vms_lib_mkarchive (bfd *abfd, enum vms_lib_kind kind)
666 struct lib_tdata *tdata;
668 tdata = (struct lib_tdata *) bfd_zalloc (abfd, sizeof (struct lib_tdata));
669 if (tdata == NULL)
670 return FALSE;
672 abfd->tdata.any = (void *)tdata;
673 vms_get_time (&tdata->credat_hi, &tdata->credat_lo);
675 tdata->kind = kind;
676 switch (kind)
678 case vms_lib_alpha:
679 tdata->ver = LBR_MAJORID;
680 tdata->mhd_size = offsetof (struct vms_mhd, pad1);
681 tdata->type = LBR__C_TYP_EOBJ;
682 break;
683 case vms_lib_ia64:
684 tdata->ver = LBR_ELFMAJORID;
685 tdata->mhd_size = sizeof (struct vms_mhd);
686 tdata->type = LBR__C_TYP_IOBJ;
687 break;
688 default:
689 abort ();
692 tdata->nbr_modules = 0;
693 tdata->artdata.symdef_count = 0;
694 tdata->modules = NULL;
695 tdata->artdata.symdefs = NULL;
696 tdata->cache = NULL;
698 return TRUE;
701 bfd_boolean
702 _bfd_vms_lib_alpha_mkarchive (bfd *abfd)
704 return _bfd_vms_lib_mkarchive (abfd, vms_lib_alpha);
707 bfd_boolean
708 _bfd_vms_lib_ia64_mkarchive (bfd *abfd)
710 return _bfd_vms_lib_mkarchive (abfd, vms_lib_ia64);
713 /* Find NAME in the symbol index. Return the index. */
715 symindex
716 _bfd_vms_lib_find_symbol (bfd *abfd, const char *name)
718 struct lib_tdata *tdata = bfd_libdata (abfd);
719 carsym *syms = tdata->artdata.symdefs;
720 int lo, hi;
722 /* Open-coded binary search for speed. */
723 lo = 0;
724 hi = tdata->artdata.symdef_count - 1;
726 while (lo <= hi)
728 int mid = lo + (hi - lo) / 2;
729 int diff;
731 diff = (char)(name[0] - syms[mid].name[0]);
732 if (diff == 0)
733 diff = strcmp (name, syms[mid].name);
734 if (diff == 0)
735 return mid;
736 else if (diff < 0)
737 hi = mid - 1;
738 else
739 lo = mid + 1;
741 return BFD_NO_MORE_SYMBOLS;
744 /* IO vector for archive member. Need that because members are not linearly
745 stored in archives. */
747 struct vms_lib_iovec
749 /* Current offset. */
750 ufile_ptr where;
752 /* Length of the module, when known. */
753 ufile_ptr file_len;
755 /* Current position in the record from bfd_bread point of view (ie, after
756 decompression). 0 means that no data byte have been read, -2 and -1
757 are reserved for the length word. */
758 int rec_pos;
759 #define REC_POS_NL -4
760 #define REC_POS_PAD -3
761 #define REC_POS_LEN0 -2
762 #define REC_POS_LEN1 -1
764 /* Record length. */
765 unsigned short rec_len;
766 /* Number of bytes to read in the current record. */
767 unsigned short rec_rem;
768 /* Offset of the next block. */
769 file_ptr next_block;
770 /* Current *data* offset in the data block. */
771 unsigned short blk_off;
773 /* Offset of the first block. Extracted from the index. */
774 file_ptr first_block;
776 /* Initial next_block. Extracted when the MHD is read. */
777 file_ptr init_next_block;
778 /* Initial blk_off, once the MHD is read. */
779 unsigned short init_blk_off;
781 /* Used to store any 3 byte record, which could be the EOF pattern. */
782 unsigned char pattern[4];
784 /* DCX. */
785 struct dcxsbm_desc *dcxsbms;
786 /* Current submap. */
787 struct dcxsbm_desc *dcx_sbm;
788 /* Current offset in the submap. */
789 unsigned int dcx_offset;
790 int dcx_pos;
792 /* Compressed buffer. */
793 unsigned char *dcx_buf;
794 /* Size of the buffer. Used to resize. */
795 unsigned int dcx_max;
796 /* Number of valid bytes in the buffer. */
797 unsigned int dcx_rlen;
800 /* Return the current position. */
802 static file_ptr
803 vms_lib_btell (struct bfd *abfd)
805 struct vms_lib_iovec *vec = (struct vms_lib_iovec *) abfd->iostream;
806 return vec->where;
809 /* Read the header of the next data block if all bytes of the current block
810 have been read. */
812 static bfd_boolean
813 vms_lib_read_block (struct bfd *abfd)
815 struct vms_lib_iovec *vec = (struct vms_lib_iovec *) abfd->iostream;
817 if (vec->blk_off == DATA__LENGTH)
819 unsigned char hdr[DATA__DATA];
821 /* Read next block. */
822 if (bfd_seek (abfd->my_archive, vec->next_block, SEEK_SET) != 0)
823 return FALSE;
824 if (bfd_bread (hdr, sizeof (hdr), abfd->my_archive) != sizeof (hdr))
825 return FALSE;
826 vec->next_block = (bfd_getl32 (hdr + 2) - 1) * VMS_BLOCK_SIZE;
827 vec->blk_off = sizeof (hdr);
829 return TRUE;
832 /* Read NBYTES from ABFD into BUF if not NULL. If BUF is NULL, bytes are
833 not stored. Read linearly from the library, but handle blocks. This
834 function does not handle records nor EOF. */
836 static file_ptr
837 vms_lib_bread_raw (struct bfd *abfd, unsigned char *buf, file_ptr nbytes)
839 struct vms_lib_iovec *vec = (struct vms_lib_iovec *) abfd->iostream;
840 file_ptr res;
842 res = 0;
843 while (nbytes > 0)
845 unsigned int l;
847 /* Be sure the current data block is read. */
848 if (!vms_lib_read_block (abfd))
849 return -1;
851 /* Do not read past the data block, do not read more than requested. */
852 l = DATA__LENGTH - vec->blk_off;
853 if (l > nbytes)
854 l = nbytes;
855 if (l == 0)
856 return 0;
857 if (buf != NULL)
859 /* Really read into BUF. */
860 if (bfd_bread (buf, l, abfd->my_archive) != l)
861 return -1;
863 else
865 /* Make as if we are reading. */
866 if (bfd_seek (abfd->my_archive, l, SEEK_CUR) != 0)
867 return -1;
870 if (buf != NULL)
871 buf += l;
872 vec->blk_off += l;
873 nbytes -= l;
874 res += l;
876 return res;
879 /* Decompress NBYTES from VEC. Store the bytes into BUF if not NULL. */
881 static file_ptr
882 vms_lib_dcx (struct vms_lib_iovec *vec, unsigned char *buf, file_ptr nbytes)
884 struct dcxsbm_desc *sbm;
885 unsigned int i;
886 unsigned int offset;
887 unsigned int j;
888 file_ptr res = 0;
890 /* The loop below expect to deliver at least one byte. */
891 if (nbytes == 0)
892 return 0;
894 /* Get the current state. */
895 sbm = vec->dcx_sbm;
896 offset = vec->dcx_offset;
897 j = vec->dcx_pos & 7;
899 for (i = vec->dcx_pos >> 3; i < vec->dcx_rlen; i++)
901 unsigned char b = vec->dcx_buf[i];
903 for (; j < 8; j++)
905 if (b & (1 << j))
906 offset++;
907 if (!(sbm->flags[offset >> 3] & (1 << (offset & 7))))
909 unsigned int n_offset = sbm->nodes[offset];
910 if (n_offset == 0)
912 /* End of buffer. Stay where we are. */
913 vec->dcx_pos = (i << 3) + j;
914 if (b & (1 << j))
915 offset--;
916 vec->dcx_offset = offset;
917 vec->dcx_sbm = sbm;
918 return res;
920 offset = 2 * n_offset;
922 else
924 unsigned char v = sbm->nodes[offset];
926 if (sbm->next != NULL)
927 sbm = vec->dcxsbms + sbm->next[v];
928 offset = 0;
929 res++;
931 if (buf)
933 *buf++ = v;
934 nbytes--;
936 if (nbytes == 0)
938 vec->dcx_pos = (i << 3) + j + 1;
939 vec->dcx_offset = offset;
940 vec->dcx_sbm = sbm;
942 return res;
947 j = 0;
949 return -1;
952 /* Standard IOVEC function. */
954 static file_ptr
955 vms_lib_bread (struct bfd *abfd, void *vbuf, file_ptr nbytes)
957 struct vms_lib_iovec *vec = (struct vms_lib_iovec *) abfd->iostream;
958 file_ptr res;
959 file_ptr chunk;
960 unsigned char *buf = (unsigned char *)vbuf;
962 /* Do not read past the end. */
963 if (vec->where >= vec->file_len)
964 return 0;
966 res = 0;
967 while (nbytes > 0)
969 if (vec->rec_rem == 0)
971 unsigned char blen[2];
973 /* Read record length. */
974 if (vms_lib_bread_raw (abfd, blen, sizeof (blen)) != sizeof (blen))
975 return -1;
976 vec->rec_len = bfd_getl16 (blen);
977 if (bfd_libdata (abfd->my_archive)->kind == vms_lib_txt)
979 /* Discard record size and align byte. */
980 vec->rec_pos = 0;
981 vec->rec_rem = vec->rec_len;
983 else
985 /* Prepend record size. */
986 vec->rec_pos = REC_POS_LEN0;
987 vec->rec_rem = (vec->rec_len + 1) & ~1; /* With align byte. */
989 if (vec->rec_len == 3)
991 /* Possibly end of file. Check the pattern. */
992 if (vms_lib_bread_raw (abfd, vec->pattern, 4) != 4)
993 return -1;
994 if (!memcmp (vec->pattern, eotdesc + 2, 3))
996 /* This is really an EOF. */
997 vec->where += res;
998 vec->file_len = vec->where;
999 return res;
1003 if (vec->dcxsbms != NULL)
1005 /* This is a compressed member. */
1006 unsigned int len;
1007 file_ptr elen;
1009 /* Be sure there is enough room for the expansion. */
1010 len = (vec->rec_len + 1) & ~1;
1011 if (len > vec->dcx_max)
1013 while (len > vec->dcx_max)
1014 vec->dcx_max *= 2;
1015 vec->dcx_buf = bfd_alloc (abfd, vec->dcx_max);
1016 if (vec->dcx_buf == NULL)
1017 return -1;
1020 /* Read the compressed record. */
1021 vec->dcx_rlen = len;
1022 if (vec->rec_len == 3)
1024 /* Already read. */
1025 memcpy (vec->dcx_buf, vec->pattern, 3);
1027 else
1029 elen = vms_lib_bread_raw (abfd, vec->dcx_buf, len);
1030 if (elen != len)
1031 return -1;
1034 /* Dummy expansion to get the expanded length. */
1035 vec->dcx_offset = 0;
1036 vec->dcx_sbm = vec->dcxsbms;
1037 vec->dcx_pos = 0;
1038 elen = vms_lib_dcx (vec, NULL, 0x10000);
1039 if (elen < 0)
1040 return -1;
1041 vec->rec_len = elen;
1042 vec->rec_rem = elen;
1044 /* Reset the state. */
1045 vec->dcx_offset = 0;
1046 vec->dcx_sbm = vec->dcxsbms;
1047 vec->dcx_pos = 0;
1050 if (vec->rec_pos < 0)
1052 unsigned char c;
1053 switch (vec->rec_pos)
1055 case REC_POS_LEN0:
1056 c = vec->rec_len & 0xff;
1057 vec->rec_pos = REC_POS_LEN1;
1058 break;
1059 case REC_POS_LEN1:
1060 c = (vec->rec_len >> 8) & 0xff;
1061 vec->rec_pos = 0;
1062 break;
1063 case REC_POS_PAD:
1064 c = 0;
1065 vec->rec_rem = 0;
1066 break;
1067 case REC_POS_NL:
1068 c = '\n';
1069 vec->rec_rem = 0;
1070 break;
1071 default:
1072 abort ();
1074 if (buf != NULL)
1076 *buf = c;
1077 buf++;
1079 nbytes--;
1080 res++;
1081 continue;
1084 if (nbytes > vec->rec_rem)
1085 chunk = vec->rec_rem;
1086 else
1087 chunk = nbytes;
1089 if (vec->dcxsbms != NULL)
1091 /* Optimize the stat() case: no need to decompress again as we
1092 know the length. */
1093 if (!(buf == NULL && chunk == vec->rec_rem))
1094 chunk = vms_lib_dcx (vec, buf, chunk);
1096 else
1098 if (vec->rec_len == 3)
1100 if (buf != NULL)
1101 memcpy (buf, vec->pattern + vec->rec_pos, chunk);
1103 else
1104 chunk = vms_lib_bread_raw (abfd, buf, chunk);
1106 if (chunk < 0)
1107 return -1;
1108 res += chunk;
1109 if (buf != NULL)
1110 buf += chunk;
1111 nbytes -= chunk;
1112 vec->rec_pos += chunk;
1113 vec->rec_rem -= chunk;
1115 if (vec->rec_rem == 0)
1117 /* End of record reached. */
1118 if (bfd_libdata (abfd->my_archive)->kind == vms_lib_txt)
1120 if ((vec->rec_len & 1) == 1
1121 && vec->rec_len != 3
1122 && vec->dcxsbms == NULL)
1124 /* Eat the pad byte. */
1125 unsigned char pad;
1126 if (vms_lib_bread_raw (abfd, &pad, 1) != 1)
1127 return -1;
1129 vec->rec_pos = REC_POS_NL;
1130 vec->rec_rem = 1;
1132 else
1134 if ((vec->rec_len & 1) == 1 && vec->dcxsbms != NULL)
1136 vec->rec_pos = REC_POS_PAD;
1137 vec->rec_rem = 1;
1142 vec->where += res;
1143 return res;
1146 /* Standard function, but we currently only handle the rewind case. */
1148 static int
1149 vms_lib_bseek (struct bfd *abfd, file_ptr offset, int whence)
1151 struct vms_lib_iovec *vec = (struct vms_lib_iovec *) abfd->iostream;
1153 if (whence == SEEK_SET && offset == 0)
1155 vec->where = 0;
1156 vec->rec_rem = 0;
1157 vec->dcx_pos = -1;
1158 vec->blk_off = vec->init_blk_off;
1159 vec->next_block = vec->init_next_block;
1161 if (bfd_seek (abfd->my_archive, vec->first_block, SEEK_SET) != 0)
1162 return -1;
1164 else
1165 abort ();
1166 return 0;
1169 static file_ptr
1170 vms_lib_bwrite (struct bfd *abfd ATTRIBUTE_UNUSED,
1171 const void *where ATTRIBUTE_UNUSED,
1172 file_ptr nbytes ATTRIBUTE_UNUSED)
1174 return -1;
1177 static int
1178 vms_lib_bclose (struct bfd *abfd)
1180 abfd->iostream = NULL;
1181 return 0;
1184 static int
1185 vms_lib_bflush (struct bfd *abfd ATTRIBUTE_UNUSED)
1187 return 0;
1190 static int
1191 vms_lib_bstat (struct bfd *abfd ATTRIBUTE_UNUSED,
1192 struct stat *sb ATTRIBUTE_UNUSED)
1194 /* Not supported. */
1195 return 0;
1198 static void *
1199 vms_lib_bmmap (struct bfd *abfd ATTRIBUTE_UNUSED,
1200 void *addr ATTRIBUTE_UNUSED,
1201 bfd_size_type len ATTRIBUTE_UNUSED,
1202 int prot ATTRIBUTE_UNUSED,
1203 int flags ATTRIBUTE_UNUSED,
1204 file_ptr offset ATTRIBUTE_UNUSED,
1205 void **map_addr ATTRIBUTE_UNUSED,
1206 bfd_size_type *map_len ATTRIBUTE_UNUSED)
1208 return (void *) -1;
1211 static const struct bfd_iovec vms_lib_iovec = {
1212 &vms_lib_bread, &vms_lib_bwrite, &vms_lib_btell, &vms_lib_bseek,
1213 &vms_lib_bclose, &vms_lib_bflush, &vms_lib_bstat, &vms_lib_bmmap
1216 /* Open a library module. FILEPOS is the position of the module header. */
1218 static bfd_boolean
1219 vms_lib_bopen (bfd *el, file_ptr filepos)
1221 struct vms_lib_iovec *vec;
1222 unsigned char buf[256];
1223 struct vms_mhd *mhd;
1224 struct lib_tdata *tdata = bfd_libdata (el->my_archive);
1225 unsigned int len;
1227 /* Allocate and initialized the iovec. */
1228 vec = bfd_zalloc (el, sizeof (*vec));
1229 if (vec == NULL)
1230 return FALSE;
1232 el->iostream = vec;
1233 el->iovec = &vms_lib_iovec;
1235 /* File length is not known. */
1236 vec->file_len = -1;
1238 /* Read the first data block. */
1239 vec->next_block = filepos & ~(VMS_BLOCK_SIZE - 1);
1240 vec->blk_off = DATA__LENGTH;
1241 if (!vms_lib_read_block (el))
1242 return FALSE;
1244 /* Prepare to read the first record. */
1245 vec->blk_off = filepos & (VMS_BLOCK_SIZE - 1);
1246 vec->rec_rem = 0;
1247 if (bfd_seek (el->my_archive, filepos, SEEK_SET) != 0)
1248 return FALSE;
1250 /* Read Record length + MHD + align byte. */
1251 len = tdata->mhd_size;
1252 if (vms_lib_bread_raw (el, buf, 2) != 2)
1253 return FALSE;
1254 if (bfd_getl16 (buf) != len)
1255 return FALSE;
1256 len = (len + 1) & ~1;
1257 BFD_ASSERT (len <= sizeof (buf));
1258 if (vms_lib_bread_raw (el, buf, len) != len)
1259 return FALSE;
1261 /* Get info from mhd. */
1262 mhd = (struct vms_mhd *)buf;
1263 /* Check id. */
1264 if (mhd->id != MHD__C_MHDID)
1265 return FALSE;
1266 if (len >= MHD__C_MHDLEN + 1)
1267 el->selective_search = (mhd->objstat & MHD__M_SELSRC) ? 1 : 0;
1268 el->mtime = vms_rawtime_to_time_t (mhd->datim);
1269 el->mtime_set = TRUE;
1271 /* Reinit the iovec so that seek() will point to the first record after
1272 the mhd. */
1273 vec->where = 0;
1274 vec->init_blk_off = vec->blk_off;
1275 vec->init_next_block = vec->next_block;
1276 vec->first_block = bfd_tell (el->my_archive);
1277 vec->dcxsbms = bfd_libdata (el->my_archive)->dcxsbm;
1279 if (vec->dcxsbms != NULL)
1281 /* Handle DCX. */
1282 vec->dcx_max = 10 * 1024;
1283 vec->dcx_buf = bfd_alloc (el, vec->dcx_max);
1284 vec->dcx_pos = -1;
1285 if (vec->dcx_buf == NULL)
1286 return -1;
1288 return TRUE;
1291 /* Get member MODIDX. Return NULL in case of error. */
1293 static bfd *
1294 _bfd_vms_lib_get_module (bfd *abfd, unsigned int modidx)
1296 struct lib_tdata *tdata = bfd_libdata (abfd);
1297 bfd *res;
1298 file_ptr file_off;
1299 char *name;
1301 /* Sanity check. */
1302 if (modidx >= tdata->nbr_modules)
1303 return NULL;
1305 /* Already loaded. */
1306 if (tdata->cache[modidx])
1307 return tdata->cache[modidx];
1309 /* Build it. */
1310 file_off = tdata->modules[modidx].file_offset;
1311 if (tdata->type != LBR__C_TYP_IOBJ)
1313 res = _bfd_create_empty_archive_element_shell (abfd);
1314 if (res == NULL)
1315 return NULL;
1317 /* Special reader to deal with data blocks. */
1318 if (!vms_lib_bopen (res, file_off))
1319 return NULL;
1321 else
1323 char buf[256];
1324 struct vms_mhd *mhd;
1325 struct areltdata *arelt;
1327 /* Sanity check. The MHD must be big enough to contain module size. */
1328 if (tdata->mhd_size < offsetof (struct vms_mhd, modsize) + 4)
1329 return NULL;
1331 /* Read the MHD now. */
1332 if (bfd_seek (abfd, file_off, SEEK_SET) != 0)
1333 return NULL;
1334 if (bfd_bread (buf, tdata->mhd_size, abfd) != tdata->mhd_size)
1335 return NULL;
1337 res = _bfd_create_empty_archive_element_shell (abfd);
1338 if (res == NULL)
1339 return NULL;
1340 arelt = bfd_zalloc (res, sizeof (*arelt));
1341 if (arelt == NULL)
1342 return NULL;
1343 res->arelt_data = arelt;
1345 /* Get info from mhd. */
1346 mhd = (struct vms_mhd *)buf;
1347 if (mhd->id != MHD__C_MHDID)
1348 return NULL;
1349 if (tdata->mhd_size >= offsetof (struct vms_mhd, objstat) + 1)
1350 res->selective_search = (mhd->objstat & MHD__M_SELSRC) ? 1 : 0;
1351 res->mtime = vms_rawtime_to_time_t (mhd->datim);
1352 res->mtime_set = TRUE;
1354 arelt->parsed_size = bfd_getl32 (mhd->modsize);
1356 /* No need for a special reader as members are stored linearly.
1357 Just skip the MHD. */
1358 res->origin = file_off + tdata->mhd_size;
1361 /* Set filename. */
1362 name = tdata->modules[modidx].name;
1363 switch (tdata->type)
1365 case LBR__C_TYP_IOBJ:
1366 case LBR__C_TYP_EOBJ:
1367 /* For object archives, append .obj to mimic standard behaviour. */
1369 size_t namelen = strlen (name);
1370 char *name1 = bfd_alloc (res, namelen + 4 + 1);
1371 memcpy (name1, name, namelen);
1372 strcpy (name1 + namelen, ".obj");
1373 name = name1;
1375 break;
1376 default:
1377 break;
1379 res->filename = name;
1381 tdata->cache[modidx] = res;
1383 return res;
1386 /* Standard function: get member at IDX. */
1388 bfd *
1389 _bfd_vms_lib_get_elt_at_index (bfd *abfd, symindex symidx)
1391 struct lib_tdata *tdata = bfd_libdata (abfd);
1392 file_ptr file_off;
1393 unsigned int modidx;
1395 /* Check symidx. */
1396 if (symidx > tdata->artdata.symdef_count)
1397 return NULL;
1398 file_off = tdata->artdata.symdefs[symidx].file_offset;
1400 /* Linear-scan. */
1401 for (modidx = 0; modidx < tdata->nbr_modules; modidx++)
1403 if (tdata->modules[modidx].file_offset == file_off)
1404 break;
1406 if (modidx >= tdata->nbr_modules)
1407 return NULL;
1409 return _bfd_vms_lib_get_module (abfd, modidx);
1412 /* Elements of an imagelib are stubs. You can get the real image with this
1413 function. */
1415 bfd *
1416 _bfd_vms_lib_get_imagelib_file (bfd *el)
1418 bfd *archive = el->my_archive;
1419 const char *modname = el->filename;
1420 int modlen = strlen (modname);
1421 char *filename;
1422 int j;
1423 bfd *res;
1425 /* Convert module name to lower case and append '.exe'. */
1426 filename = bfd_alloc (el, modlen + 5);
1427 if (filename == NULL)
1428 return NULL;
1429 for (j = 0; j < modlen; j++)
1430 if (ISALPHA (modname[j]))
1431 filename[j] = TOLOWER (modname[j]);
1432 else
1433 filename[j] = modname[j];
1434 memcpy (filename + modlen, ".exe", 5);
1436 filename = _bfd_append_relative_path (archive, filename);
1437 if (filename == NULL)
1438 return NULL;
1439 res = bfd_openr (filename, NULL);
1441 if (res == NULL)
1443 (*_bfd_error_handler)(_("could not open shared image '%s' from '%s'"),
1444 filename, archive->filename);
1445 bfd_release (archive, filename);
1446 return NULL;
1449 /* FIXME: put it in a cache ? */
1450 return res;
1453 /* Standard function. */
1455 bfd *
1456 _bfd_vms_lib_openr_next_archived_file (bfd *archive,
1457 bfd *last_file)
1459 unsigned int idx;
1460 bfd *res;
1462 if (!last_file)
1463 idx = 0;
1464 else
1465 idx = last_file->proxy_origin + 1;
1467 if (idx >= bfd_libdata (archive)->nbr_modules)
1469 bfd_set_error (bfd_error_no_more_archived_files);
1470 return NULL;
1473 res = _bfd_vms_lib_get_module (archive, idx);
1474 if (res == NULL)
1475 return res;
1476 res->proxy_origin = idx;
1477 return res;
1480 /* Standard function. Just compute the length. */
1483 _bfd_vms_lib_generic_stat_arch_elt (bfd *abfd, struct stat *st)
1485 struct lib_tdata *tdata;
1487 /* Sanity check. */
1488 if (abfd->my_archive == NULL)
1490 bfd_set_error (bfd_error_invalid_operation);
1491 return -1;
1494 tdata = bfd_libdata (abfd->my_archive);
1495 if (tdata->type != LBR__C_TYP_IOBJ)
1497 struct vms_lib_iovec *vec = (struct vms_lib_iovec *) abfd->iostream;
1499 if (vec->file_len == (ufile_ptr)-1)
1501 if (vms_lib_bseek (abfd, 0, SEEK_SET) != 0)
1502 return -1;
1504 /* Compute length. */
1505 while (vms_lib_bread (abfd, NULL, 1 << 20) > 0)
1508 st->st_size = vec->file_len;
1510 else
1512 st->st_size = ((struct areltdata *)abfd->arelt_data)->parsed_size;
1515 if (abfd->mtime_set)
1516 st->st_mtime = abfd->mtime;
1517 else
1518 st->st_mtime = 0;
1519 st->st_uid = 0;
1520 st->st_gid = 0;
1521 st->st_mode = 0644;
1523 return 0;
1526 /* Internal representation of an index entry. */
1528 struct lib_index
1530 /* Corresponding archive member. */
1531 bfd *abfd;
1533 /* Number of reference to this entry. */
1534 unsigned int ref;
1536 /* Length of the key. */
1537 unsigned short namlen;
1539 /* Key. */
1540 const char *name;
1543 /* Used to sort index entries. */
1545 static int
1546 lib_index_cmp (const void *lv, const void *rv)
1548 const struct lib_index *l = lv;
1549 const struct lib_index *r = rv;
1551 return strcmp (l->name, r->name);
1554 /* Maximum number of index blocks level. */
1556 #define MAX_LEVEL 10
1558 /* Get the size of an index entry. */
1560 static unsigned int
1561 get_idxlen (struct lib_index *idx, bfd_boolean is_elfidx)
1563 if (is_elfidx)
1565 /* 9 is the size of struct vms_elfidx without keyname. */
1566 if (idx->namlen > MAX_KEYLEN)
1567 return 9 + sizeof (struct vms_kbn);
1568 else
1569 return 9 + idx->namlen;
1571 else
1573 /* 7 is the size of struct vms_idx without keyname. */
1574 return 7 + idx->namlen;
1578 /* Write the index composed by NBR symbols contained in IDX.
1579 VBN is the first vbn to be used, and will contain on return the last vbn.
1580 Can be called with ABFD set to NULL just to size the index.
1581 If not null, TOPVBN will be assigned to the vbn of the root index tree.
1582 IS_ELFIDX is true for elfidx (ie ia64) indexes layout.
1583 Return TRUE on success. */
1585 static bfd_boolean
1586 vms_write_index (bfd *abfd,
1587 struct lib_index *idx, unsigned int nbr, unsigned int *vbn,
1588 unsigned int *topvbn, bfd_boolean is_elfidx)
1590 /* The index is organized as a tree. This function implements a naive
1591 algorithm to balance the tree: it fills the leaves, and create a new
1592 branch when all upper leaves and branches are full. We only keep in
1593 memory a path to the current leaf. */
1594 unsigned int i;
1595 int j;
1596 int level;
1597 /* Disk blocks for the current path. */
1598 struct vms_indexdef *rblk[MAX_LEVEL];
1599 /* Info on the current blocks. */
1600 struct idxblk
1602 unsigned int vbn; /* VBN of the block. */
1603 /* The last entry is identified so that it could be copied to the
1604 parent block. */
1605 unsigned short len; /* Length up to the last entry. */
1606 unsigned short lastlen; /* Length of the last entry. */
1607 } blk[MAX_LEVEL];
1609 /* The kbn blocks are used to store long symbol names. */
1610 unsigned int kbn_sz = 0; /* Number of bytes available in the kbn block. */
1611 unsigned int kbn_vbn = 0; /* VBN of the kbn block. */
1612 unsigned char *kbn_blk = NULL; /* Contents of the kbn block. */
1614 if (nbr == 0)
1616 /* No entries. Very easy to handle. */
1617 if (topvbn != NULL)
1618 *topvbn = 0;
1619 return TRUE;
1622 if (abfd == NULL)
1624 /* Sort the index the first time this function is called. */
1625 qsort (idx, nbr, sizeof (struct lib_index), lib_index_cmp);
1628 /* Allocate first index block. */
1629 level = 1;
1630 if (abfd != NULL)
1631 rblk[0] = bfd_zmalloc (sizeof (struct vms_indexdef));
1632 blk[0].vbn = (*vbn)++;
1633 blk[0].len = 0;
1634 blk[0].lastlen = 0;
1636 for (i = 0; i < nbr; i++, idx++)
1638 unsigned int idxlen;
1639 int flush = 0;
1640 unsigned int key_vbn = 0;
1641 unsigned int key_off = 0;
1643 idxlen = get_idxlen (idx, is_elfidx);
1645 if (is_elfidx && idx->namlen > MAX_KEYLEN)
1647 /* If the key (ie name) is too long, write it in the kbn block. */
1648 unsigned int kl = idx->namlen;
1649 unsigned int kl_chunk;
1650 const char *key = idx->name;
1652 /* Write the key in the kbn, chunk after chunk. */
1655 if (kbn_sz < sizeof (struct vms_kbn))
1657 /* Not enough room in the kbn block. */
1658 if (abfd != NULL)
1660 /* Write it to the disk (if there is one). */
1661 if (kbn_vbn != 0)
1663 if (vms_write_block (abfd, kbn_vbn, kbn_blk) != TRUE)
1664 return FALSE;
1666 else
1668 kbn_blk = bfd_malloc (VMS_BLOCK_SIZE);
1669 if (kbn_blk == NULL)
1670 return FALSE;
1672 *(unsigned short *)kbn_blk = 0;
1674 /* Allocate a new block for the keys. */
1675 kbn_vbn = (*vbn)++;
1676 kbn_sz = VMS_BLOCK_SIZE - 2;
1678 /* Size of the chunk written to the current key block. */
1679 if (kl + sizeof (struct vms_kbn) > kbn_sz)
1680 kl_chunk = kbn_sz - sizeof (struct vms_kbn);
1681 else
1682 kl_chunk = kl;
1684 if (kbn_blk != NULL)
1686 struct vms_kbn *kbn;
1688 kbn = (struct vms_kbn *)(kbn_blk + VMS_BLOCK_SIZE - kbn_sz);
1690 if (key_vbn == 0)
1692 /* Save the rfa of the first chunk. */
1693 key_vbn = kbn_vbn;
1694 key_off = VMS_BLOCK_SIZE - kbn_sz;
1697 bfd_putl16 (kl_chunk, kbn->keylen);
1698 if (kl_chunk == kl)
1700 /* No next chunk. */
1701 bfd_putl32 (0, kbn->rfa.vbn);
1702 bfd_putl16 (0, kbn->rfa.offset);
1704 else
1706 /* Next chunk will be at the start of the next block. */
1707 bfd_putl32 (*vbn, kbn->rfa.vbn);
1708 bfd_putl16 (2, kbn->rfa.offset);
1710 memcpy ((char *)(kbn + 1), key, kl_chunk);
1711 key += kl_chunk;
1713 kl -= kl_chunk;
1714 kl_chunk = (kl_chunk + 1) & ~1; /* Always align. */
1715 kbn_sz -= kl_chunk + sizeof (struct vms_kbn);
1717 while (kl > 0);
1720 /* Check if a block might overflow. In this case we will flush this
1721 block and all the blocks below it. */
1722 for (j = 0; j < level; j++)
1723 if (blk[j].len + blk[j].lastlen + idxlen > INDEXDEF__BLKSIZ)
1724 flush = j + 1;
1726 for (j = 0; j < level; j++)
1728 if (j < flush)
1730 /* There is not enough room to write the new entry in this
1731 block or in a parent block. */
1733 if (j + 1 == level)
1735 BFD_ASSERT (level < MAX_LEVEL);
1737 /* Need to create a parent. */
1738 if (abfd != NULL)
1740 rblk[level] = bfd_zmalloc (sizeof (struct vms_indexdef));
1741 bfd_putl32 (*vbn, rblk[j]->parent);
1743 blk[level].vbn = (*vbn)++;
1744 blk[level].len = 0;
1745 blk[level].lastlen = blk[j].lastlen;
1747 level++;
1750 /* Update parent block: write the last entry from the current
1751 block. */
1752 if (abfd != NULL)
1754 struct vms_rfa *rfa;
1756 /* Pointer to the last entry in parent block. */
1757 rfa = (struct vms_rfa *)(rblk[j + 1]->keys + blk[j + 1].len);
1759 /* Copy the whole entry. */
1760 BFD_ASSERT (blk[j + 1].lastlen == blk[j].lastlen);
1761 memcpy (rfa, rblk[j]->keys + blk[j].len, blk[j].lastlen);
1762 /* Fix the entry (which in always the first field of an
1763 entry. */
1764 bfd_putl32 (blk[j].vbn, rfa->vbn);
1765 bfd_putl16 (RFADEF__C_INDEX, rfa->offset);
1768 if (j + 1 == flush)
1770 /* And allocate it. Do it only on the block that won't be
1771 flushed (so that the parent of the parent can be
1772 updated too). */
1773 blk[j + 1].len += blk[j + 1].lastlen;
1774 blk[j + 1].lastlen = 0;
1777 /* Write this block on the disk. */
1778 if (abfd != NULL)
1780 bfd_putl16 (blk[j].len + blk[j].lastlen, rblk[j]->used);
1781 if (vms_write_block (abfd, blk[j].vbn, rblk[j]) != TRUE)
1782 return FALSE;
1785 /* Reset this block. */
1786 blk[j].len = 0;
1787 blk[j].lastlen = 0;
1788 blk[j].vbn = (*vbn)++;
1791 /* Append it to the block. */
1792 if (j == 0)
1794 /* Keep the previous last entry. */
1795 blk[j].len += blk[j].lastlen;
1797 if (abfd != NULL)
1799 struct vms_rfa *rfa;
1801 rfa = (struct vms_rfa *)(rblk[j]->keys + blk[j].len);
1802 bfd_putl32 ((idx->abfd->proxy_origin / VMS_BLOCK_SIZE) + 1,
1803 rfa->vbn);
1804 bfd_putl16
1805 ((idx->abfd->proxy_origin % VMS_BLOCK_SIZE)
1806 + (is_elfidx ? 0 : DATA__DATA),
1807 rfa->offset);
1809 if (is_elfidx)
1811 /* Use elfidx format. */
1812 struct vms_elfidx *en = (struct vms_elfidx *)rfa;
1814 en->flags = 0;
1815 if (key_vbn != 0)
1817 /* Long symbol name. */
1818 struct vms_kbn *k = (struct vms_kbn *)(en->keyname);
1819 bfd_putl16 (sizeof (struct vms_kbn), en->keylen);
1820 bfd_putl16 (idx->namlen, k->keylen);
1821 bfd_putl32 (key_vbn, k->rfa.vbn);
1822 bfd_putl16 (key_off, k->rfa.offset);
1823 en->flags |= ELFIDX__SYMESC;
1825 else
1827 bfd_putl16 (idx->namlen, en->keylen);
1828 memcpy (en->keyname, idx->name, idx->namlen);
1831 else
1833 /* Use idx format. */
1834 struct vms_idx *en = (struct vms_idx *)rfa;
1835 en->keylen = idx->namlen;
1836 memcpy (en->keyname, idx->name, idx->namlen);
1840 /* The last added key can now be the last one all blocks in the
1841 path. */
1842 blk[j].lastlen = idxlen;
1846 /* Save VBN of the root. */
1847 if (topvbn != NULL)
1848 *topvbn = blk[level - 1].vbn;
1850 if (abfd == NULL)
1851 return TRUE;
1853 /* Flush. */
1854 for (j = 1; j < level; j++)
1856 /* Update parent block: write the new entry. */
1857 unsigned char *en;
1858 unsigned char *par;
1859 struct vms_rfa *rfa;
1861 en = rblk[j - 1]->keys + blk[j - 1].len;
1862 par = rblk[j]->keys + blk[j].len;
1863 BFD_ASSERT (blk[j].lastlen == blk[j - 1].lastlen);
1864 memcpy (par, en, blk[j - 1].lastlen);
1865 rfa = (struct vms_rfa *)par;
1866 bfd_putl32 (blk[j - 1].vbn, rfa->vbn);
1867 bfd_putl16 (RFADEF__C_INDEX, rfa->offset);
1870 for (j = 0; j < level; j++)
1872 /* Write this block on the disk. */
1873 bfd_putl16 (blk[j].len + blk[j].lastlen, rblk[j]->used);
1874 if (vms_write_block (abfd, blk[j].vbn, rblk[j]) != TRUE)
1875 return FALSE;
1877 free (rblk[j]);
1880 /* Write the last kbn (if any). */
1881 if (kbn_vbn != 0)
1883 if (vms_write_block (abfd, kbn_vbn, kbn_blk) != TRUE)
1884 return FALSE;
1885 free (kbn_blk);
1888 return TRUE;
1891 /* Append data to the data block DATA. Force write if PAD is true. */
1893 static bfd_boolean
1894 vms_write_data_block (bfd *arch, struct vms_datadef *data, file_ptr *off,
1895 const unsigned char *buf, unsigned int len, int pad)
1897 while (len > 0 || pad)
1899 unsigned int doff = *off & (VMS_BLOCK_SIZE - 1);
1900 unsigned int remlen = (DATA__LENGTH - DATA__DATA) - doff;
1901 unsigned int l;
1903 l = (len > remlen) ? remlen : len;
1904 memcpy (data->data + doff, buf, l);
1905 buf += l;
1906 len -= l;
1907 doff += l;
1908 *off += l;
1910 if (doff == (DATA__LENGTH - DATA__DATA) || (len == 0 && pad))
1912 data->recs = 0;
1913 data->fill_1 = 0;
1914 bfd_putl32 ((*off / VMS_BLOCK_SIZE) + 2, data->link);
1916 if (bfd_bwrite (data, sizeof (*data), arch) != sizeof (*data))
1917 return FALSE;
1919 *off += DATA__LENGTH - doff;
1921 if (len == 0)
1922 break;
1925 return TRUE;
1928 /* Build the symbols index. */
1930 static bfd_boolean
1931 _bfd_vms_lib_build_map (unsigned int nbr_modules,
1932 struct lib_index *modules,
1933 unsigned int *res_cnt,
1934 struct lib_index **res)
1936 unsigned int i;
1937 asymbol **syms = NULL;
1938 long syms_max = 0;
1939 struct lib_index *map = NULL;
1940 unsigned int map_max = 1024; /* Fine initial default. */
1941 unsigned int map_count = 0;
1943 map = (struct lib_index *) bfd_malloc (map_max * sizeof (struct lib_index));
1944 if (map == NULL)
1945 goto error_return;
1947 /* Gather symbols. */
1948 for (i = 0; i < nbr_modules; i++)
1950 long storage;
1951 long symcount;
1952 long src_count;
1953 bfd *current = modules[i].abfd;
1955 if ((bfd_get_file_flags (current) & HAS_SYMS) == 0)
1956 continue;
1958 storage = bfd_get_symtab_upper_bound (current);
1959 if (storage < 0)
1960 goto error_return;
1962 if (storage != 0)
1964 if (storage > syms_max)
1966 if (syms_max > 0)
1967 free (syms);
1968 syms_max = storage;
1969 syms = (asymbol **) bfd_malloc (syms_max);
1970 if (syms == NULL)
1971 goto error_return;
1973 symcount = bfd_canonicalize_symtab (current, syms);
1974 if (symcount < 0)
1975 goto error_return;
1977 /* Now map over all the symbols, picking out the ones we
1978 want. */
1979 for (src_count = 0; src_count < symcount; src_count++)
1981 flagword flags = (syms[src_count])->flags;
1982 asection *sec = syms[src_count]->section;
1984 if ((flags & BSF_GLOBAL
1985 || flags & BSF_WEAK
1986 || flags & BSF_INDIRECT
1987 || bfd_is_com_section (sec))
1988 && ! bfd_is_und_section (sec))
1990 struct lib_index *new_map;
1992 /* This symbol will go into the archive header. */
1993 if (map_count == map_max)
1995 map_max *= 2;
1996 new_map = (struct lib_index *)
1997 bfd_realloc (map, map_max * sizeof (struct lib_index));
1998 if (new_map == NULL)
1999 goto error_return;
2000 map = new_map;
2003 map[map_count].abfd = current;
2004 map[map_count].namlen = strlen (syms[src_count]->name);
2005 map[map_count].name = syms[src_count]->name;
2006 map_count++;
2007 modules[i].ref++;
2013 *res_cnt = map_count;
2014 *res = map;
2015 return TRUE;
2017 error_return:
2018 if (syms_max > 0)
2019 free (syms);
2020 if (map != NULL)
2021 free (map);
2022 return FALSE;
2025 /* Do the hard work: write an archive on the disk. */
2027 bfd_boolean
2028 _bfd_vms_lib_write_archive_contents (bfd *arch)
2030 bfd *current;
2031 unsigned int nbr_modules;
2032 struct lib_index *modules;
2033 unsigned int nbr_symbols;
2034 struct lib_index *symbols;
2035 struct lib_tdata *tdata = bfd_libdata (arch);
2036 unsigned int i;
2037 file_ptr off;
2038 unsigned int nbr_mod_iblk;
2039 unsigned int nbr_sym_iblk;
2040 unsigned int vbn;
2041 unsigned int mod_idx_vbn;
2042 unsigned int sym_idx_vbn;
2043 bfd_boolean is_elfidx = tdata->kind == vms_lib_ia64;
2044 unsigned int max_keylen = is_elfidx ? MAX_EKEYLEN : MAX_KEYLEN;
2046 /* Count the number of modules (and do a first sanity check). */
2047 nbr_modules = 0;
2048 for (current = arch->archive_head;
2049 current != NULL;
2050 current = current->archive_next)
2052 /* This check is checking the bfds for the objects we're reading
2053 from (which are usually either an object file or archive on
2054 disk), not the archive entries we're writing to. We don't
2055 actually create bfds for the archive members, we just copy
2056 them byte-wise when we write out the archive. */
2057 if (bfd_write_p (current) || !bfd_check_format (current, bfd_object))
2059 bfd_set_error (bfd_error_invalid_operation);
2060 goto input_err;
2063 nbr_modules++;
2066 /* Build the modules list. */
2067 BFD_ASSERT (tdata->modules == NULL);
2068 modules = bfd_alloc (arch, nbr_modules * sizeof (struct lib_index));
2069 if (modules == NULL)
2070 return FALSE;
2072 for (current = arch->archive_head, i = 0;
2073 current != NULL;
2074 current = current->archive_next, i++)
2076 unsigned int nl;
2078 modules[i].abfd = current;
2079 modules[i].name = vms_get_module_name (current->filename, FALSE);
2080 modules[i].ref = 1;
2082 /* FIXME: silently truncate long names ? */
2083 nl = strlen (modules[i].name);
2084 modules[i].namlen = (nl > max_keylen ? max_keylen : nl);
2087 /* Create the module index. */
2088 vbn = 0;
2089 if (!vms_write_index (NULL, modules, nbr_modules, &vbn, NULL, is_elfidx))
2090 return FALSE;
2091 nbr_mod_iblk = vbn;
2093 /* Create symbol index. */
2094 if (!_bfd_vms_lib_build_map (nbr_modules, modules, &nbr_symbols, &symbols))
2095 return FALSE;
2097 vbn = 0;
2098 if (!vms_write_index (NULL, symbols, nbr_symbols, &vbn, NULL, is_elfidx))
2099 return FALSE;
2100 nbr_sym_iblk = vbn;
2102 /* Write modules and remember their position. */
2103 off = (1 + nbr_mod_iblk + nbr_sym_iblk) * VMS_BLOCK_SIZE;
2105 if (bfd_seek (arch, off, SEEK_SET) != 0)
2106 return FALSE;
2108 for (i = 0; i < nbr_modules; i++)
2110 struct vms_datadef data;
2111 unsigned char blk[VMS_BLOCK_SIZE];
2112 struct vms_mhd *mhd;
2113 unsigned int sz;
2115 current = modules[i].abfd;
2116 current->proxy_origin = off;
2118 if (is_elfidx)
2119 sz = 0;
2120 else
2122 /* Write the MHD as a record (ie, size first). */
2123 sz = 2;
2124 bfd_putl16 (tdata->mhd_size, blk);
2126 mhd = (struct vms_mhd *)(blk + sz);
2127 memset (mhd, 0, sizeof (struct vms_mhd));
2128 mhd->lbrflag = 0;
2129 mhd->id = MHD__C_MHDID;
2130 mhd->objidlng = 4;
2131 memcpy (mhd->objid, "V1.0", 4);
2132 bfd_putl32 (modules[i].ref, mhd->refcnt);
2133 /* FIXME: datim. */
2135 sz += tdata->mhd_size;
2136 sz = (sz + 1) & ~1;
2138 /* Rewind the member to be put into the archive. */
2139 if (bfd_seek (current, 0, SEEK_SET) != 0)
2140 goto input_err;
2142 /* Copy the member into the archive. */
2143 if (is_elfidx)
2145 unsigned int modsize = 0;
2146 bfd_size_type amt;
2147 file_ptr off_hdr = off;
2149 /* Read to complete the first block. */
2150 amt = bfd_bread (blk + sz, VMS_BLOCK_SIZE - sz, current);
2151 if (amt == (bfd_size_type)-1)
2152 goto input_err;
2153 modsize = amt;
2154 if (amt < VMS_BLOCK_SIZE - sz)
2156 /* The member size is less than a block. Pad the block. */
2157 memset (blk + sz + amt, 0, VMS_BLOCK_SIZE - sz - amt);
2159 bfd_putl32 (modsize, mhd->modsize);
2161 /* Write the first block (which contains an mhd). */
2162 if (bfd_bwrite (blk, VMS_BLOCK_SIZE, arch) != VMS_BLOCK_SIZE)
2163 goto input_err;
2164 off += VMS_BLOCK_SIZE;
2166 if (amt == VMS_BLOCK_SIZE - sz)
2168 /* Copy the remaining. */
2169 char buffer[DEFAULT_BUFFERSIZE];
2171 while (1)
2173 amt = bfd_bread (buffer, sizeof (buffer), current);
2174 if (amt == (bfd_size_type)-1)
2175 goto input_err;
2176 if (amt == 0)
2177 break;
2178 modsize += amt;
2179 if (amt != sizeof (buffer))
2181 /* Clear the padding. */
2182 memset (buffer + amt, 0, sizeof (buffer) - amt);
2183 amt = (amt + VMS_BLOCK_SIZE) & ~(VMS_BLOCK_SIZE - 1);
2185 if (bfd_bwrite (buffer, amt, arch) != amt)
2186 goto input_err;
2187 off += amt;
2190 /* Now that the size is known, write the first block (again). */
2191 bfd_putl32 (modsize, mhd->modsize);
2192 if (bfd_seek (arch, off_hdr, SEEK_SET) != 0
2193 || bfd_bwrite (blk, VMS_BLOCK_SIZE, arch) != VMS_BLOCK_SIZE)
2194 goto input_err;
2195 if (bfd_seek (arch, off, SEEK_SET) != 0)
2196 goto input_err;
2199 else
2201 /* Write the MHD. */
2202 if (vms_write_data_block (arch, &data, &off, blk, sz, 0) < 0)
2203 goto input_err;
2205 /* Write the member. */
2206 while (1)
2208 sz = bfd_bread (blk, sizeof (blk), current);
2209 if (sz == 0)
2210 break;
2211 if (vms_write_data_block (arch, &data, &off, blk, sz, 0) < 0)
2212 goto input_err;
2215 /* Write the end of module marker. */
2216 if (vms_write_data_block (arch, &data, &off,
2217 eotdesc, sizeof (eotdesc), 1) < 0)
2218 goto input_err;
2222 /* Write the indexes. */
2223 vbn = 2;
2224 if (vms_write_index (arch, modules, nbr_modules, &vbn, &mod_idx_vbn,
2225 is_elfidx) != TRUE)
2226 return FALSE;
2227 if (vms_write_index (arch, symbols, nbr_symbols, &vbn, &sym_idx_vbn,
2228 is_elfidx) != TRUE)
2229 return FALSE;
2231 /* Write libary header. */
2233 unsigned char blk[VMS_BLOCK_SIZE];
2234 struct vms_lhd *lhd = (struct vms_lhd *)blk;
2235 struct vms_idd *idd = (struct vms_idd *)(blk + sizeof (*lhd));
2236 unsigned int idd_flags;
2237 unsigned int saneid;
2239 memset (blk, 0, sizeof (blk));
2241 lhd->type = tdata->type;
2242 lhd->nindex = 2;
2243 switch (tdata->kind)
2245 case vms_lib_alpha:
2246 saneid = LHD_SANEID3;
2247 break;
2248 case vms_lib_ia64:
2249 saneid = LHD_SANEID6;
2250 break;
2251 default:
2252 abort ();
2254 bfd_putl32 (saneid, lhd->sanity);
2255 bfd_putl16 (tdata->ver, lhd->majorid);
2256 bfd_putl16 (0, lhd->minorid);
2257 snprintf ((char *)lhd->lbrver + 1, sizeof (lhd->lbrver) - 1,
2258 "GNU ar %u.%u.%u",
2259 (unsigned)(BFD_VERSION / 100000000UL),
2260 (unsigned)(BFD_VERSION / 1000000UL) % 100,
2261 (unsigned)(BFD_VERSION / 10000UL) % 100);
2262 lhd->lbrver[sizeof (lhd->lbrver) - 1] = 0;
2263 lhd->lbrver[0] = strlen ((char *)lhd->lbrver + 1);
2265 bfd_putl32 (tdata->credat_lo, lhd->credat + 0);
2266 bfd_putl32 (tdata->credat_hi, lhd->credat + 4);
2267 vms_raw_get_time (lhd->updtim);
2269 lhd->mhdusz = tdata->mhd_size - MHD__C_USRDAT;
2271 bfd_putl32 (nbr_modules + nbr_symbols, lhd->idxcnt);
2272 bfd_putl32 (nbr_modules, lhd->modcnt);
2273 bfd_putl32 (nbr_modules, lhd->modhdrs);
2275 /* Number of blocks for index. */
2276 bfd_putl32 (nbr_mod_iblk + nbr_sym_iblk, lhd->idxblks);
2277 bfd_putl32 (vbn - 1, lhd->hipreal);
2278 bfd_putl32 (vbn - 1, lhd->hiprusd);
2280 /* VBN of the next free block. */
2281 bfd_putl32 ((off / VMS_BLOCK_SIZE) + 1, lhd->nextvbn);
2282 bfd_putl32 ((off / VMS_BLOCK_SIZE) + 1, lhd->nextrfa + 0);
2283 bfd_putl16 (0, lhd->nextrfa + 4);
2285 /* First index (modules name). */
2286 idd_flags = IDD__FLAGS_ASCII | IDD__FLAGS_VARLENIDX
2287 | IDD__FLAGS_NOCASECMP | IDD__FLAGS_NOCASENTR;
2288 bfd_putl16 (idd_flags, idd->flags);
2289 bfd_putl16 (max_keylen + 1, idd->keylen);
2290 bfd_putl16 (mod_idx_vbn, idd->vbn);
2291 idd++;
2293 /* Second index (symbols name). */
2294 bfd_putl16 (idd_flags, idd->flags);
2295 bfd_putl16 (max_keylen + 1, idd->keylen);
2296 bfd_putl16 (sym_idx_vbn, idd->vbn);
2297 idd++;
2299 if (vms_write_block (arch, 1, blk) != TRUE)
2300 return FALSE;
2303 return TRUE;
2305 input_err:
2306 bfd_set_error (bfd_error_on_input, current, bfd_get_error ());
2307 return FALSE;
2310 /* Add a target for text library. This costs almost nothing and is useful to
2311 read VMS library on the host. */
2313 const bfd_target vms_lib_txt_vec =
2315 "vms-libtxt", /* Name. */
2316 bfd_target_unknown_flavour,
2317 BFD_ENDIAN_UNKNOWN, /* byteorder */
2318 BFD_ENDIAN_UNKNOWN, /* header_byteorder */
2319 0, /* Object flags. */
2320 0, /* Sect flags. */
2321 0, /* symbol_leading_char. */
2322 ' ', /* ar_pad_char. */
2323 15, /* ar_max_namelen. */
2324 0, /* match priority. */
2325 bfd_getl64, bfd_getl_signed_64, bfd_putl64,
2326 bfd_getl32, bfd_getl_signed_32, bfd_putl32,
2327 bfd_getl16, bfd_getl_signed_16, bfd_putl16,
2328 bfd_getl64, bfd_getl_signed_64, bfd_putl64,
2329 bfd_getl32, bfd_getl_signed_32, bfd_putl32,
2330 bfd_getl16, bfd_getl_signed_16, bfd_putl16,
2332 {_bfd_dummy_target, _bfd_dummy_target, /* bfd_check_format. */
2333 _bfd_vms_lib_txt_archive_p, _bfd_dummy_target},
2334 {bfd_false, bfd_false, bfd_false, bfd_false}, /* bfd_set_format. */
2335 {bfd_false, bfd_false, bfd_false, bfd_false}, /* bfd_write_contents. */
2337 BFD_JUMP_TABLE_GENERIC (_bfd_generic),
2338 BFD_JUMP_TABLE_COPY (_bfd_generic),
2339 BFD_JUMP_TABLE_CORE (_bfd_nocore),
2340 BFD_JUMP_TABLE_ARCHIVE (_bfd_vms_lib),
2341 BFD_JUMP_TABLE_SYMBOLS (_bfd_nosymbols),
2342 BFD_JUMP_TABLE_RELOCS (_bfd_norelocs),
2343 BFD_JUMP_TABLE_WRITE (_bfd_nowrite),
2344 BFD_JUMP_TABLE_LINK (_bfd_nolink),
2345 BFD_JUMP_TABLE_DYNAMIC (_bfd_nodynamic),
2347 NULL,
2349 NULL