1 /* outieee.c output routines for the Netwide Assembler to produce
2 * IEEE-std object files
4 * The Netwide Assembler is copyright (C) 1996 Simon Tatham and
5 * Julian Hall. All rights reserved. The software is
6 * redistributable under the licence given in the file "Licence"
7 * distributed in the NASM archive.
10 /* notes: I have tried to make this correspond to the IEEE version
11 * of the standard, specifically the primary ASCII version. It should
12 * be trivial to create the binary version given this source (which is
13 * one of MANY things that have to be done to make this correspond to
14 * the hp-microtek version of the standard).
16 * 16-bit support is assumed to use 24-bit addresses
17 * The linker can sort out segmentation-specific stuff
18 * if it keeps track of externals
19 * in terms of being relative to section bases
21 * A non-standard variable type, the 'Yn' variable, has been introduced.
22 * Basically it is a reference to extern 'n'- denoting the low limit
23 * (L-variable) of the section that extern 'n' is defined in. Like the
24 * x variable, there may be no explicit assignment to it, it is derived
25 * from the public definition corresponding to the extern name. This
26 * is required because the one thing the mufom guys forgot to do well was
27 * take into account segmented architectures.
29 * I use comment classes for various things and these are undefined by
32 * Debug info should be considered totally non-standard (local labels are
33 * standard but linenum records are not covered by the standard.
34 * Type defs have the standard format but absolute meanings for ordinal
35 * types are not covered by the standard.)
37 * David Lindauer, LADsoft
43 #include <stdarg.h> /* Note: we need the ANSI version of stdarg.h */
54 static char ieee_infile
[FILENAME_MAX
];
55 static int ieee_uppercase
;
58 static ldfunc deflabel
;
63 #define HUNKSIZE 1024 /* Size of the data hunk */
64 #define EXT_BLKSIZ 512
65 #define LDPERLINE 32 /* bytes per line in output */
70 struct LineNumber
*next
;
71 struct ieeeSection
*segment
;
76 static struct FileName
{
77 struct FileName
*next
;
86 } *arrhead
, **arrtail
;
88 static struct ieeePublic
{
89 struct ieeePublic
*next
;
92 long segment
; /* only if it's far-absolute */
94 int type
; /* for debug purposes */
95 } *fpubhead
, **fpubtail
, *last_defined
;
97 static struct ieeeExternal
{
98 struct ieeeExternal
*next
;
101 } *exthead
, **exttail
;
103 static int externals
;
105 static struct ExtBack
{
106 struct ExtBack
*next
;
107 int index
[EXT_BLKSIZ
];
110 /* NOTE: the first segment MUST be the lineno segment */
111 static struct ieeeSection
{
112 struct ieeeObjData
*data
,*datacurr
;
113 struct ieeeSection
*next
;
114 struct ieeeFixupp
*fptr
, * flptr
;
115 long index
; /* the NASM segment id */
116 long ieee_index
; /* the OBJ-file segment index */
118 long align
; /* can be SEG_ABS + absolute addr */
125 long use32
; /* is this segment 32-bit? */
126 struct ieeePublic
*pubhead
, **pubtail
, *lochead
, **loctail
;
128 } *seghead
, **segtail
, *ieee_seg_needs_update
;
131 struct ieeeObjData
*next
;
132 unsigned char data
[HUNKSIZE
];
136 struct ieeeFixupp
*next
;
154 static long ieee_entry_seg
, ieee_entry_ofs
;
157 extern struct ofmt of_ieee
;
159 static void ieee_data_new(struct ieeeSection
*);
160 static void ieee_write_fixup (long, long, struct ieeeSection
*,
161 int, unsigned long, long);
162 static void ieee_install_fixup(struct ieeeSection
*, struct ieeeFixupp
*);
163 static long ieee_segment (char *, int, int *);
164 static void ieee_write_file(int debuginfo
);
165 static void ieee_write_byte(struct ieeeSection
*, int);
166 static void ieee_write_word(struct ieeeSection
*, int);
167 static void ieee_write_dword(struct ieeeSection
*, long);
168 static void ieee_putascii(char *, ...);
169 static void ieee_putcs(int);
170 static long ieee_putld(long, long, unsigned char *);
171 static long ieee_putlr(struct ieeeFixupp
*);
172 static void ieee_unqualified_name(char *, char *);
178 static void ieee_init (FILE *fp
, efunc errfunc
, ldfunc ldef
, evalfunc eval
)
186 fpubtail
= &fpubhead
;
192 seghead
= ieee_seg_needs_update
= NULL
;
194 ieee_entry_seg
= NO_SEG
;
195 ieee_uppercase
= FALSE
;
197 of_ieee
.current_dfmt
->init (&of_ieee
,NULL
,fp
,errfunc
);
199 static int ieee_set_info(enum geninfo type
, char **val
)
209 static void ieee_cleanup (int debuginfo
)
211 ieee_write_file(debuginfo
);
212 of_ieee
.current_dfmt
->cleanup ();
215 struct ieeeSection
*segtmp
= seghead
;
216 seghead
= seghead
->next
;
217 while (segtmp
->pubhead
) {
218 struct ieeePublic
*pubtmp
= segtmp
->pubhead
;
219 segtmp
->pubhead
= pubtmp
->next
;
222 while (segtmp
->fptr
) {
223 struct ieeeFixupp
*fixtmp
= segtmp
->fptr
;
224 segtmp
->fptr
= fixtmp
->next
;
227 while (segtmp
->data
) {
228 struct ieeeObjData
*dattmp
= segtmp
->data
;
229 segtmp
->data
= dattmp
->next
;
235 struct ieeePublic
*pubtmp
= fpubhead
;
236 fpubhead
= fpubhead
->next
;
240 struct ieeeExternal
*exttmp
= exthead
;
241 exthead
= exthead
->next
;
245 struct ExtBack
*ebtmp
= ebhead
;
246 ebhead
= ebhead
->next
;
251 * callback for labels
253 static void ieee_deflabel (char *name
, long segment
,
254 long offset
, int is_global
, char *special
) {
256 * We have three cases:
258 * (i) `segment' is a segment-base. If so, set the name field
259 * for the segment structure it refers to, and then
262 * (ii) `segment' is one of our segments, or a SEG_ABS segment.
263 * Save the label position for later output of a PUBDEF record.
266 * (iii) `segment' is not one of our segments. Save the label
267 * position for later output of an EXTDEF.
269 struct ieeeExternal
*ext
;
271 struct ieeeSection
*seg
;
275 error(ERR_NONFATAL
, "unrecognised symbol type `%s'", special
);
278 * First check for the double-period, signifying something
281 if (name
[0] == '.' && name
[1] == '.') {
282 if (!strcmp(name
, "..start")) {
283 ieee_entry_seg
= segment
;
284 ieee_entry_ofs
= offset
;
292 if (ieee_seg_needs_update
) {
293 ieee_seg_needs_update
->name
= name
;
296 if (segment
< SEG_ABS
&& segment
!= NO_SEG
&& segment
% 2)
302 if (segment
>= SEG_ABS
) {
304 * SEG_ABS subcase of (ii).
307 struct ieeePublic
*pub
;
309 pub
= *fpubtail
= nasm_malloc(sizeof(*pub
));
310 fpubtail
= &pub
->next
;
313 pub
->offset
= offset
;
314 pub
->segment
= segment
& ~SEG_ABS
;
319 for (seg
= seghead
; seg
&& is_global
; seg
= seg
->next
)
320 if (seg
->index
== segment
) {
321 struct ieeePublic
*pub
;
323 last_defined
= pub
= *seg
->pubtail
= nasm_malloc(sizeof(*pub
));
324 seg
->pubtail
= &pub
->next
;
327 pub
->offset
= offset
;
328 pub
->index
= seg
->ieee_index
;
337 ext
= *exttail
= nasm_malloc(sizeof(*ext
));
339 exttail
= &ext
->next
;
342 ext
->commonsize
= offset
;
348 eb
= *ebtail
= nasm_malloc(sizeof(*eb
));
352 while (i
> EXT_BLKSIZ
) {
356 eb
= *ebtail
= nasm_malloc(sizeof(*eb
));
362 eb
->index
[i
] = externals
++;
370 static void ieee_out (long segto
, void *data
, unsigned long type
,
371 long segment
, long wrt
) {
373 unsigned long realtype
;
374 unsigned char *ucdata
;
376 struct ieeeSection
*seg
;
379 * handle absolute-assembly (structure definitions)
381 if (segto
== NO_SEG
) {
382 if ((type
& OUT_TYPMASK
) != OUT_RESERVE
)
383 error (ERR_NONFATAL
, "attempt to assemble code in [ABSOLUTE]"
389 * If `any_segs' is still FALSE, we must define a default
393 int tempint
; /* ignored */
394 if (segto
!= ieee_segment("__NASMDEFSEG", 2, &tempint
))
395 error (ERR_PANIC
, "strange segment conditions in IEEE driver");
399 * Find the segment we are targetting.
401 for (seg
= seghead
; seg
; seg
= seg
->next
)
402 if (seg
->index
== segto
)
405 error (ERR_PANIC
, "code directed to nonexistent segment?");
407 size
= type
& OUT_SIZMASK
;
408 realtype
= type
& OUT_TYPMASK
;
409 if (realtype
== OUT_RAWDATA
) {
412 ieee_write_byte(seg
,*ucdata
++);
413 } else if (realtype
== OUT_ADDRESS
|| realtype
== OUT_REL2ADR
||
414 realtype
== OUT_REL4ADR
) {
415 if (segment
== NO_SEG
&& realtype
!= OUT_ADDRESS
)
416 error(ERR_NONFATAL
, "relative call to absolute address not"
417 " supported by IEEE format");
418 ldata
= *(long *)data
;
419 if (realtype
== OUT_REL2ADR
)
421 if (realtype
== OUT_REL4ADR
)
423 ieee_write_fixup (segment
, wrt
, seg
, size
, realtype
,ldata
);
425 ieee_write_word (seg
, ldata
);
427 ieee_write_dword (seg
, ldata
);
429 else if (realtype
== OUT_RESERVE
) {
431 ieee_write_byte(seg
,0);
435 static void ieee_data_new(struct ieeeSection
*segto
) {
438 segto
->data
= segto
->datacurr
= nasm_malloc(sizeof(*(segto
->datacurr
)));
440 segto
->datacurr
= segto
->datacurr
->next
= nasm_malloc(sizeof(*(segto
->datacurr
)));
441 segto
->datacurr
->next
= NULL
;
446 * this routine is unalduterated bloatware. I usually don't do this
447 * but I might as well see what it is like on a harmless program.
448 * If anyone wants to optimize this is a good canditate!
450 static void ieee_write_fixup (long segment
, long wrt
, struct ieeeSection
* segto
,
451 int size
, unsigned long realtype
, long offset
) {
452 struct ieeeSection
*target
;
455 /* Don't put a fixup for things NASM can calculate */
456 if (wrt
== NO_SEG
&& segment
== NO_SEG
)
460 /* if it is a WRT offset */
465 s
.id1
= -(wrt
-SEG_ABS
);
467 if (wrt
%2 && realtype
!= OUT_REL2ADR
&& realtype
!= OUT_REL4ADR
) {
470 for (target
= seghead
; target
; target
= target
->next
)
471 if (target
->index
== wrt
)
474 s
.id1
= target
->ieee_index
;
475 for (target
= seghead
; target
; target
= target
->next
)
476 if (target
->index
== segment
)
480 s
.id2
= target
->ieee_index
;
483 * Now we assume the segment field is being used
484 * to hold an extern index
487 struct ExtBack
*eb
= ebhead
;
488 while (i
> EXT_BLKSIZ
) {
495 /* if we have an extern decide the type and make a record
500 s
.id2
= eb
->index
[i
];
504 "Source of WRT must be an offset");
510 "unrecognised WRT value in ieee_write_fixup");
513 error(ERR_NONFATAL
,"target of WRT must be a section ");
516 ieee_install_fixup(segto
,&s
);
519 /* Pure segment fixup ? */
520 if (segment
!= NO_SEG
) {
523 if (segment
>= SEG_ABS
) {
524 /* absolute far segment fixup */
525 s
.id1
= -(segment
-~SEG_ABS
);
527 else if (segment
% 2) {
528 /* fixup to named segment */
530 for (target
= seghead
; target
; target
= target
->next
)
531 if (target
->index
== segment
-1)
534 s
.id1
= target
->ieee_index
;
537 * Now we assume the segment field is being used
538 * to hold an extern index
541 struct ExtBack
*eb
= ebhead
;
542 while (i
> EXT_BLKSIZ
) {
549 /* if we have an extern decide the type and make a record
552 if (realtype
== OUT_REL2ADR
|| realtype
== OUT_REL4ADR
) {
553 error(ERR_PANIC
,"Segment of a rel not supported in ieee_write_fixup");
556 /* If we want the segment */
559 s
.id1
= eb
->index
[i
];
564 /* If we get here the seg value doesn't make sense */
566 "unrecognised segment value in ieee_write_fixup");
570 /* Assume we are offsetting directly from a section
571 * So look up the target segment
573 for (target
= seghead
; target
; target
= target
->next
)
574 if (target
->index
== segment
)
577 if (realtype
== OUT_REL2ADR
|| realtype
== OUT_REL4ADR
) {
578 /* PC rel to a known offset */
579 s
.id1
= target
->ieee_index
;
585 /* We were offsetting from a seg */
586 s
.id1
= target
->ieee_index
;
594 * Now we assume the segment field is being used
595 * to hold an extern index
598 struct ExtBack
*eb
= ebhead
;
599 while (i
> EXT_BLKSIZ
) {
606 /* if we have an extern decide the type and make a record
609 if (realtype
== OUT_REL2ADR
|| realtype
== OUT_REL4ADR
) {
612 s
.id1
= eb
->index
[i
];
615 /* else we want the external offset */
618 s
.id1
= eb
->index
[i
];
623 /* If we get here the seg value doesn't make sense */
625 "unrecognised segment value in ieee_write_fixup");
628 if (size
!= 2 && s
.ftype
== FT_SEG
)
629 error(ERR_NONFATAL
, "IEEE format can only handle 2-byte"
630 " segment base references");
632 ieee_install_fixup(segto
,&s
);
635 /* should never get here */
637 static void ieee_install_fixup(struct ieeeSection
*seg
, struct ieeeFixupp
*fix
)
639 struct ieeeFixupp
*f
;
640 f
= nasm_malloc(sizeof(struct ieeeFixupp
));
641 memcpy(f
,fix
,sizeof(struct ieeeFixupp
));
642 f
->offset
= seg
->currentpos
;
643 seg
->currentpos
+= fix
->size
;
646 seg
->flptr
= seg
->flptr
->next
= f
;
648 seg
->fptr
= seg
->flptr
= f
;
655 static long ieee_segment (char *name
, int pass
, int *bits
) {
657 * We call the label manager here to define a name for the new
658 * segment, and when our _own_ label-definition stub gets
659 * called in return, it should register the new segment name
660 * using the pointer it gets passed. That way we save memory,
661 * by sponging off the label manager.
667 return seghead
->index
;
669 struct ieeeSection
*seg
;
670 int ieee_idx
, attrs
, rn_error
;
674 * Look for segment attributes.
678 name
++; /* hack, but a documented one */
680 while (*p
&& !isspace(*p
))
684 while (*p
&& isspace(*p
))
688 while (*p
&& !isspace(*p
))
692 while (*p
&& isspace(*p
))
700 for (seg
= seghead
; seg
; seg
= seg
->next
) {
702 if (!strcmp(seg
->name
, name
)) {
703 if (attrs
> 0 && pass
== 1)
704 error(ERR_WARNING
, "segment attributes specified on"
705 " redeclaration of segment: ignoring");
714 *segtail
= seg
= nasm_malloc(sizeof(*seg
));
716 segtail
= &seg
->next
;
717 seg
->index
= seg_alloc();
718 seg
->ieee_index
= ieee_idx
;
722 seg
->align
= 1; /* default */
723 seg
->use32
= *bits
== 32; /* default to user spec */
724 seg
->combine
= CMB_PUBLIC
; /* default */
726 seg
->pubtail
= &seg
->pubhead
;
730 seg
->loctail
= &seg
->lochead
;
733 * Process the segment attributes.
741 * `p' contains a segment attribute.
743 if (!nasm_stricmp(p
, "private"))
744 seg
->combine
= CMB_PRIVATE
;
745 else if (!nasm_stricmp(p
, "public"))
746 seg
->combine
= CMB_PUBLIC
;
747 else if (!nasm_stricmp(p
, "common"))
748 seg
->combine
= CMB_COMMON
;
749 else if (!nasm_stricmp(p
, "use16"))
751 else if (!nasm_stricmp(p
, "use32"))
753 else if (!nasm_strnicmp(p
, "align=", 6)) {
754 seg
->align
= readnum(p
+6, &rn_error
);
759 error (ERR_NONFATAL
, "segment alignment should be"
762 switch ((int) seg
->align
) {
774 error(ERR_NONFATAL
, "invalid alignment value %d",
779 } else if (!nasm_strnicmp(p
, "absolute=", 9)) {
780 seg
->align
= SEG_ABS
+ readnum(p
+9, &rn_error
);
782 error (ERR_NONFATAL
, "argument to `absolute' segment"
783 " attribute should be numeric");
787 ieee_seg_needs_update
= seg
;
788 if (seg
->align
>= SEG_ABS
)
789 deflabel (name
, NO_SEG
, seg
->align
- SEG_ABS
,
790 NULL
, FALSE
, FALSE
, &of_ieee
, error
);
792 deflabel (name
, seg
->index
+1, 0L,
793 NULL
, FALSE
, FALSE
, &of_ieee
, error
);
794 ieee_seg_needs_update
= NULL
;
804 * directives supported
806 static int ieee_directive (char *directive
, char *value
, int pass
)
811 if (!strcmp(directive
, "uppercase")) {
812 ieee_uppercase
= TRUE
;
818 * Return segment data
820 static long ieee_segbase (long segment
)
822 struct ieeeSection
*seg
;
825 * Find the segment in our list.
827 for (seg
= seghead
; seg
; seg
= seg
->next
)
828 if (seg
->index
== segment
-1)
832 return segment
; /* not one of ours - leave it alone */
834 if (seg
->align
>= SEG_ABS
)
835 return seg
->align
; /* absolute segment */
837 return segment
; /* no special treatment */
842 static void ieee_filename (char *inname
, char *outname
, efunc error
) {
843 strcpy(ieee_infile
, inname
);
844 standard_extension (inname
, outname
, ".o", error
);
847 static void ieee_write_file (int debuginfo
) {
851 struct ieeeSection
*seg
;
852 struct ieeePublic
*pub
, *loc
;
853 struct ieeeExternal
*ext
;
854 struct ieeeObjData
*data
;
855 struct ieeeFixupp
*fix
;
857 static char boast
[] = "The Netwide Assembler " NASM_VER
;
861 * Write the module header
863 ieee_putascii("MBFNASM,%02X%s.\r\n",strlen(ieee_infile
),ieee_infile
);
866 * Write the NASM boast comment.
868 ieee_putascii("CO0,%02X%s.\r\n",strlen(boast
),boast
);
871 * write processor-specific information
873 ieee_putascii("AD8,4,L.\r\n");
879 thetime
= localtime(&reltime
);
880 ieee_putascii("DT%04d%02d%02d%02d%02d%02d.\r\n",
881 1900+thetime
->tm_year
,thetime
->tm_mon
+1,thetime
->tm_mday
,
882 thetime
->tm_hour
, thetime
->tm_min
, thetime
->tm_sec
);
884 * if debugging, dump file names
886 for (fn
= fnhead
; fn
&& debuginfo
; fn
= fn
->next
) {
887 ieee_putascii("C0105,%02X%s.\r\n",strlen(fn
->name
),fn
->name
);
890 ieee_putascii("CO101,07ENDHEAD.\r\n");
892 * the standard doesn't specify when to put checksums,
893 * we'll just do it periodically.
898 * Write the section headers
901 if (!debuginfo
&& !strcmp(seg
->name
,"??LINE"))
906 switch(seg
->combine
) {
918 ieee_unqualified_name(buf
,seg
->name
);
919 if (seg
->align
>= SEG_ABS
) {
920 ieee_putascii("ST%X,A,%02X%s.\r\n",seg
->ieee_index
,strlen(buf
), buf
);
921 ieee_putascii("ASL%X,%lX.\r\n",seg
->ieee_index
, (seg
->align
- SEG_ABS
)*16);
924 ieee_putascii("ST%X,%c,%02X%s.\r\n",seg
->ieee_index
,attrib
,strlen(buf
), buf
);
925 ieee_putascii("SA%X,%lX.\r\n",seg
->ieee_index
,seg
->align
);
926 ieee_putascii("ASS%X,%X.\r\n",seg
->ieee_index
, seg
->currentpos
);
931 * write the start address if there is one
933 if (ieee_entry_seg
) {
934 for (seg
= seghead
; seg
; seg
= seg
->next
)
935 if (seg
->index
== ieee_entry_seg
)
938 error(ERR_PANIC
,"Start address records are incorrect");
940 ieee_putascii("ASG,R%X,%lX,+.\r\n",seg
->ieee_index
, ieee_entry_ofs
);
949 for (seg
= seghead
; seg
; seg
= seg
->next
) {
950 for (pub
= seg
->pubhead
; pub
; pub
= pub
->next
) {
952 ieee_unqualified_name(buf
,pub
->name
);
953 ieee_putascii("NI%X,%02X%s.\r\n",i
, strlen(buf
), buf
);
954 if (pub
->segment
== -1)
955 ieee_putascii("ASI%X,R%X,%lX,+.\r\n", i
, pub
->index
,pub
->offset
);
957 ieee_putascii("ASI%X,%lX,%lX,+.\r\n", i
, pub
->segment
*16,pub
->offset
);
959 if (pub
->type
>= 0x100)
960 ieee_putascii("ATI%X,T%X.\r\n", i
, pub
->type
- 0x100);
962 ieee_putascii("ATI%X,%X.\r\n", i
, pub
->type
);
971 ieee_unqualified_name(buf
,pub
->name
);
972 ieee_putascii("NI%X,%02X%s.\r\n",i
, strlen(buf
), buf
);
973 if (pub
->segment
== -1)
974 ieee_putascii("ASI%X,R%X,%lX,+.\r\n", i
, pub
->index
,pub
->offset
);
976 ieee_putascii("ASI%X,%lX,%lX,+.\r\n", i
, pub
->segment
*16,pub
->offset
);
978 if (pub
->type
>= 0x100)
979 ieee_putascii("ATI%X,T%X.\r\n", i
, pub
->type
- 0x100);
981 ieee_putascii("ATI%X,%X.\r\n", i
, pub
->type
);
987 * Write the externals
993 ieee_unqualified_name(buf
,ext
->name
);
994 ieee_putascii("NX%X,%02X%s.\r\n",i
++, strlen(buf
), buf
);
1000 * IEEE doesn't have a standard pass break record
1001 * so use the ladsoft variant
1003 ieee_putascii("CO100,06ENDSYM.\r\n");
1009 for (arr
= arrhead
; arr
&& debuginfo
; arr
= arr
->next
) {
1010 ieee_putascii("TY%X,20,%X,%lX.\r\n",i
++,arr
->basetype
,arr
->size
);
1016 for (seg
= seghead
; seg
&& debuginfo
; seg
= seg
->next
) {
1017 for (loc
= seg
->lochead
; loc
; loc
= loc
->next
) {
1019 ieee_unqualified_name(buf
,loc
->name
);
1020 ieee_putascii("NN%X,%02X%s.\r\n",i
, strlen(buf
), buf
);
1021 if (loc
->segment
== -1)
1022 ieee_putascii("ASN%X,R%X,%lX,+.\r\n", i
, loc
->index
,loc
->offset
);
1024 ieee_putascii("ASN%X,%lX,%lX,+.\r\n", i
, loc
->segment
*16,loc
->offset
);
1026 if (loc
->type
>= 0x100)
1027 ieee_putascii("ATN%X,T%X.\r\n", i
, loc
->type
- 0x100);
1029 ieee_putascii("ATN%X,%X.\r\n", i
, loc
->type
);
1036 * put out section data;
1039 if (!debuginfo
&& !strcmp(seg
->name
,"??LINE"))
1042 if (seg
->currentpos
) {
1045 ieee_putascii("SB%X.\r\n",seg
->ieee_index
);
1048 size
= HUNKSIZE
- (org
%HUNKSIZE
);
1049 size
= size
+org
> seg
->currentpos
? seg
->currentpos
-org
: size
;
1050 size
= fix
->offset
- org
> size
? size
: fix
->offset
-org
;
1051 org
= ieee_putld(org
,org
+size
,data
->data
);
1052 if (org
% HUNKSIZE
== 0)
1054 if (org
== fix
->offset
) {
1055 org
+= ieee_putlr(fix
);
1059 while (org
< seg
->currentpos
&& data
) {
1060 size
= seg
->currentpos
-org
> HUNKSIZE
? HUNKSIZE
: seg
->currentpos
- org
;
1061 org
= ieee_putld(org
,org
+size
,data
->data
);
1072 ieee_putascii("ME.\r\n");
1075 static void ieee_write_byte(struct ieeeSection
*seg
, int data
) {
1077 if (!(temp
= seg
->currentpos
++ % HUNKSIZE
))
1079 seg
->datacurr
->data
[temp
] = data
;
1082 static void ieee_write_word(struct ieeeSection
*seg
, int data
) {
1083 ieee_write_byte(seg
, data
& 0xFF);
1084 ieee_write_byte(seg
,(data
>> 8) & 0xFF);
1087 static void ieee_write_dword(struct ieeeSection
*seg
, long data
) {
1088 ieee_write_byte(seg
, data
& 0xFF);
1089 ieee_write_byte(seg
,(data
>> 8) & 0xFF);
1090 ieee_write_byte(seg
,(data
>> 16) & 0xFF);
1091 ieee_write_byte(seg
,(data
>> 24) & 0xFF);
1093 static void ieee_putascii(char *format
, ...)
1099 va_start(ap
, format
);
1100 vsprintf(buffer
, format
, ap
);
1102 for (i
=0; i
< l
; i
++)
1103 if ((buffer
[i
] & 0xff) > 31)
1104 checksum
+=buffer
[i
];
1106 fprintf(ofp
,buffer
);
1110 * put out a checksum record */
1111 static void ieee_putcs(int toclear
)
1114 ieee_putascii("CS.\r\n");
1119 ieee_putascii("CS%02X.\r\n",checksum
& 127);
1124 static long ieee_putld(long start
, long end
, unsigned char *buf
)
1129 val
= start
% HUNKSIZE
;
1130 /* fill up multiple lines */
1131 while (end
- start
>= LDPERLINE
) {
1133 ieee_putascii("LD");
1134 for (i
=0; i
< LDPERLINE
; i
++) {
1135 ieee_putascii("%02X",buf
[val
++]);
1138 ieee_putascii(".\r\n");
1140 /* if no partial lines */
1143 /* make a partial line */
1144 ieee_putascii("LD");
1145 while (start
< end
) {
1146 ieee_putascii("%02X",buf
[val
++]);
1149 ieee_putascii(".\r\n");
1152 static long ieee_putlr(struct ieeeFixupp
*p
)
1155 * To deal with the vagaries of segmentation the LADsoft linker
1156 * defines two types of segments: absolute and virtual. Note that
1157 * 'absolute' in this context is a different thing from the IEEE
1158 * definition of an absolute segment type, which is also supported. If a
1159 * sement is linked in virtual mode the low limit (L-var) is
1160 * subtracted from each R,X, and P variable which appears in an
1161 * expression, so that we can have relative offsets. Meanwhile
1162 * in the ABSOLUTE mode this subtraction is not done and
1163 * so we can use absolute offsets from 0. In the LADsoft linker
1164 * this configuration is not done in the assemblker source but in
1165 * a source the linker reads. Generally this type of thing only
1166 * becomes an issue if real mode code is used. A pure 32-bit linker could
1167 * get away without defining the virtual mode...
1174 sprintf(buf
,"%lX",-p
->id1
);
1176 sprintf(buf
,"L%lX,10,/",p
->id1
);
1179 sprintf(buf
,"R%lX,%lX,+",p
->id1
,p
->addend
);
1182 sprintf(buf
,"R%lX,%lX,+,P,-,%X,-",p
->id1
,p
->addend
,p
->size
);
1187 sprintf(buf
,"R%lX,%lX,+,L%lX,+,%lX,-",p
->id2
,p
->addend
,p
->id2
,-p
->id1
*16);
1189 sprintf(buf
,"R%lX,%lX,+,L%lX,+,L%lX,-",p
->id2
,p
->addend
,p
->id2
,p
->id1
);
1192 sprintf(buf
,"X%lX",p
->id1
);
1195 sprintf(buf
,"X%lX,P,-,%lX,-",p
->id1
,size
);
1198 /* We needed a non-ieee hack here.
1199 * We introduce the Y variable, which is the low
1200 * limit of the native segment the extern resides in
1202 sprintf(buf
,"Y%lX,10,/",p
->id1
);
1206 sprintf(buf
,"X%lX,Y%lX,+,%lX,-",p
->id2
,p
->id2
,-p
->id1
*16);
1208 sprintf(buf
,"X%lX,Y%lX,+,L%lX,-",p
->id2
,p
->id2
,p
->id1
);
1211 ieee_putascii("LR(%s,%lX).\r\n", buf
, size
);
1215 /* Dump all segment data (text and fixups )*/
1217 static void ieee_unqualified_name(char *dest
, char *source
)
1219 if (ieee_uppercase
) {
1221 *dest
++ = toupper(*source
++);
1224 else strcpy(dest
,source
);
1226 void dbgls_init(struct ofmt
* of
, void * id
, FILE * fp
, efunc error
)
1236 arrindex
= ARRAY_BOT
;
1239 ieee_segment("??LINE", 2, &tempint
);
1242 static void dbgls_cleanup(void)
1244 struct ieeeSection
*segtmp
;
1246 struct FileName
*fntemp
= fnhead
;
1247 fnhead
= fnhead
->next
;
1248 nasm_free (fntemp
->name
);
1251 for (segtmp
= seghead
; segtmp
; segtmp
= segtmp
->next
) {
1252 while (segtmp
->lochead
) {
1253 struct ieeePublic
*loctmp
= segtmp
->lochead
;
1254 segtmp
->lochead
= loctmp
->next
;
1255 nasm_free (loctmp
->name
);
1260 struct Array
*arrtmp
= arrhead
;
1261 arrhead
= arrhead
->next
;
1267 * because this routine is not bracketed in
1268 * the main program, this routine will be called even if there
1269 * is no request for debug info
1270 * so, we have to make sure the ??LINE segment is avaialbe
1271 * as the first segment when this debug format is selected
1273 static void dbgls_linnum (const char *lnfname
, long lineno
, long segto
)
1275 struct FileName
*fn
;
1276 struct ieeeSection
*seg
;
1278 if (segto
== NO_SEG
)
1282 * If `any_segs' is still FALSE, we must define a default
1286 int tempint
; /* ignored */
1287 if (segto
!= ieee_segment("__NASMDEFSEG", 2, &tempint
))
1288 error (ERR_PANIC
, "strange segment conditions in OBJ driver");
1292 * Find the segment we are targetting.
1294 for (seg
= seghead
; seg
; seg
= seg
->next
)
1295 if (seg
->index
== segto
)
1298 error (ERR_PANIC
, "lineno directed to nonexistent segment?");
1300 for (fn
= fnhead
; fn
; fn
= fnhead
->next
) {
1301 if (!nasm_stricmp(lnfname
,fn
->name
))
1306 fn
= nasm_malloc ( sizeof( *fn
));
1307 fn
->name
= nasm_malloc ( strlen(lnfname
) + 1) ;
1309 strcpy (fn
->name
,lnfname
);
1314 ieee_write_byte(seghead
, fn
->index
);
1315 ieee_write_word(seghead
, lineno
);
1316 ieee_write_fixup (segto
, NO_SEG
, seghead
, 4,OUT_ADDRESS
,seg
->currentpos
);
1319 static void dbgls_deflabel (char *name
, long segment
,
1320 long offset
, int is_global
, char *special
)
1322 struct ieeeSection
*seg
;
1323 int used_special
; /* have we used the special text? */
1325 /* Keep compiler from warning about special and used_special */
1326 used_special
= special
? FALSE
: FALSE
;
1329 * If it's a special-retry from pass two, discard it.
1335 * First check for the double-period, signifying something
1338 if (name
[0] == '.' && name
[1] == '.' && name
[2] != '@') {
1345 if (ieee_seg_needs_update
)
1347 if (segment
< SEG_ABS
&& segment
!= NO_SEG
&& segment
% 2)
1350 if (segment
>= SEG_ABS
|| segment
== NO_SEG
) {
1355 * If `any_segs' is still FALSE, we might need to define a
1356 * default segment, if they're trying to declare a label in
1357 * `first_seg'. But the label should exist due to a prior
1358 * call to ieee_deflabel so we can skip that.
1361 for (seg
= seghead
; seg
; seg
= seg
->next
)
1362 if (seg
->index
== segment
) {
1363 struct ieeePublic
*loc
;
1365 * Case (ii). Maybe MODPUB someday?
1368 last_defined
= loc
= nasm_malloc (sizeof(*loc
));
1369 *seg
->loctail
= loc
;
1370 seg
->loctail
= &loc
->next
;
1372 loc
->name
= nasm_strdup(name
);
1373 loc
->offset
= offset
;
1375 loc
->index
= seg
->ieee_index
;
1379 static void dbgls_typevalue (long type
)
1381 int elem
= TYM_ELEMENTS(type
);
1382 type
= TYM_TYPE(type
);
1389 last_defined
->type
= 1; /* unsigned char */
1392 last_defined
->type
= 3; /* unsigned word */
1395 last_defined
->type
= 5; /* unsigned dword */
1398 last_defined
->type
= 9; /* float */
1401 last_defined
->type
= 10; /* qword */
1404 last_defined
->type
= 11; /* TBYTE */
1407 last_defined
->type
= 0x10; /* near label */
1412 struct Array
*arrtmp
= nasm_malloc (sizeof(*arrtmp
));
1413 int vtype
= last_defined
->type
;
1414 arrtmp
->size
= elem
;
1415 arrtmp
->basetype
= vtype
;
1416 arrtmp
->next
= NULL
;
1417 last_defined
->type
= arrindex
++ + 0x100;
1419 arrtail
= & (arrtmp
->next
);
1421 last_defined
= NULL
;
1423 static void dbgls_output (int output_type
, void *param
)
1428 static struct dfmt ladsoft_debug_form
= {
1429 "LADsoft Debug Records",
1439 static struct dfmt
*ladsoft_debug_arr
[3] = {
1440 &ladsoft_debug_form
,
1444 struct ofmt of_ieee
= {
1445 "IEEE-695 (LADsoft variant) object file format",
1462 #endif /* OF_IEEE */