Discussion:
loader
(too old to reply)
Paul Edwards
2024-02-20 08:10:55 UTC
Permalink
Ok, so here is the "hello world" program I am starting with:

D:\devel\pdos\xxx>hexdump os2test.exe
000000 4D5A7C00 01000000 04004100 FFFF0400 MZ|.......A.....
000010 00040000 00000000 40000000 00000000 ***@.......
000020 00000000 00000000 00000000 00000000 ................
000030 00000000 00000000 00000000 80000000 ................
000040 0E1FB409 BA1000CD 21B001B4 4CCD2100 ........!...L.!.
000050 54686973 2070726F 6772616D 206E6565 This program nee
000060 6473204F 532F3220 322E3020 6F722065 ds OS/2 2.0 or e
000070 71756976 616C656E 740D0A24 00000000 quivalent..$....
000080 4C580000 00000000 02000100 00000000 LX..............
000090 00020000 02000000 01000000 00000000 ................
0000A0 03000000 00080000 00100000 00000000 ................
0000B0 23000000 00000000 64000000 00000000 #.......d.......
0000C0 C4000000 03000000 0C010000 00000000 ................
0000D0 1C010000 00000000 1C010000 27010000 ............'...
0000E0 00000000 00000000 28010000 34010000 ........(...4...
0000F0 41010000 01000000 4A010000 00000000 A.......J.......
000100 CC010000 00000000 00000000 00000000 ................
000110 00000000 02000000 00000000 00000000 ................
000120 00000000 00000000 00000000 00080000 ................
000130 00000000 00000000 00000000 00000000 ................
000140 00000000 28000000 00000100 05200000 ....(........ ..
000150 01000000 01000000 00000000 0C000000 ................
000160 00000200 03200000 02000000 01000000 ..... ..........
000170 00000000 00080000 00000300 03200000 ............. ..
000180 03000000 00000000 00000000 00000000 ................
000190 27000000 27000000 0A000000 076F7332 '...'........os2
0001A0 74657374 00000000 00000000 0D000000 test............
0001B0 0D000000 08011000 011A0108 811C0001 ................
0001C0 EA08444F 5343414C 4C530000 83EC0489 ..DOSCALLS......
0001D0 E0506A09 68000002 006A01E8 00000000 .Pj.h....j......
0001E0 83C4106A 006A01E8 00000000 83C40883 ...j.j..........
0001F0 C404C368 690D0A61 62630D0A 00 ...hi..abc...

D:\devel\pdos\xxx>



Looking at it and the LX documentation I have:

000000 4D5A7C00 01000000 04004100 FFFF0400 MZ|.......A.....
000010 00040000 00000000 40000000 00000000 ***@.......
000020 00000000 00000000 00000000 00000000 ................
000030 00000000 00000000 00000000 80000000 ................
000040 0E1FB409 BA1000CD 21B001B4 4CCD2100 ........!...L.!.
000050 54686973 2070726F 6772616D 206E6565 This program nee
000060 6473204F 532F3220 322E3020 6F722065 ds OS/2 2.0 or e
000070 71756976 616C656E 740D0A24 00000000 quivalent..$....

MSDOS header (optional) ends here.

That x'40' at 0x18 is apparently important,
and the x'80' at 0x3c, which I think is
referenced as 0x1c from 0x20 means that the
extension starts at 0x80.

000080 4C580000 00000000 02000100 00000000 LX..............
000090 00020000 02000000 01000000 00000000 ................
0000A0 03000000 00080000 00100000 00000000 ................
0000B0 23000000 00000000 64000000 00000000 #.......d.......
0000C0 C4000000 03000000 0C010000 00000000 ................
0000D0 1C010000 00000000 1C010000 27010000 ............'...
0000E0 00000000 00000000 28010000 34010000 ........(...4...
0000F0 41010000 01000000 4A010000 00000000 A.......J.......
000100 CC010000 00000000 00000000 00000000 ................
000110 00000000 02000000 00000000 00000000 ................
000120 00000000 00000000 00000000 00080000 ................

And this is the end of the LX header. The last
field is x'0800' which is the stack size.

Which is indeed what I put:

.stack 800h

000130 00000000 00000000 00000000 00000000 ................
000140 00000000 28000000 00000100 05200000 ....(........ ..
000150 01000000 01000000 00000000 0C000000 ................
000160 00000200 03200000 02000000 01000000 ..... ..........
000170 00000000 00080000 00000300 03200000 ............. ..
000180 03000000 00000000 00000000 00000000 ................
000190 27000000 27000000 0A000000 076F7332 '...'........os2
0001A0 74657374 00000000 00000000 0D000000 test............
0001B0 0D000000 08011000 011A0108 811C0001 ................

My code:

Segment: _TEXT PARA USE32 00000027 bytes
0000 _os2test:
0000 83 EC 04 sub esp,0x00000004
0003 89 E0 mov eax,esp
0005 50 push eax
0006 6A 09 push 0x00000009
0008 68 00 00 00 00 push offset L$1
000D 6A 01 push 0x00000001
000F E8 00 00 00 00 call DosWrite
0014 83 C4 10 add esp,0x00000010
0017 6A 00 push 0x00000000
0019 6A 01 push 0x00000001
001B E8 00 00 00 00 call DosExit
0020 83 C4 08 add esp,0x00000008
0023 83 C4 04 add esp,0x00000004
0026 C3 ret

starts at 0x1cc

0001C0 EA08444F 5343414C 4C530000 83EC0489 ..DOSCALLS......
0001D0 E0506A09 68000002 006A01E8 00000000 .Pj.h....j......
0001E0 83C4106A 006A01E8 00000000 83C40883 ...j.j..........
0001F0 C404C368 690D0A61 62630D0A 00 ...hi..abc...

and the last byte is at 0x1f2. And then this starts immediately:

Segment: CONST DWORD USE32 0000000A bytes
0000 L$1:
0000 68 69 0D 0A 61 62 63 0D 0A 00 hi..abc...

and the executable terminates with no further ado.

So that means I am missing an interpretation of this:

000130 00000000 00000000 00000000 00000000 ................
000140 00000000 28000000 00000100 05200000 ....(........ ..
000150 01000000 01000000 00000000 0C000000 ................
000160 00000200 03200000 02000000 01000000 ..... ..........
000170 00000000 00080000 00000300 03200000 ............. ..
000180 03000000 00000000 00000000 00000000 ................
000190 27000000 27000000 0A000000 076F7332 '...'........os2
0001A0 74657374 00000000 00000000 0D000000 test............
0001B0 0D000000 08011000 011A0108 811C0001 ................
0001C0 EA08444F 5343414C 4C530000 ..DOSCALLS..

And I can see the DLL "doscalls" being referenced,
and there should be two accesses to it.

++DosWrite.DOSCALLS.DosWrite.282
++DosExit.DOSCALLS.DosExit.234

282 is x'11a'
234 is x'ea'

I can see that first at 0x1b9.

And the second is at 0x1c0, but it is followed by
x'08' which I wouldn't have expected. Maybe
the x'08' is a terminator, allowing a variable
length of DLL ordinal. I would have instead
expected a 2-byte fixed size for the DLL ordinal.

Given that DOSCALLS immediately follows the
ordinals, we need to go back to offset 0x1bc to
find something interesting, which would be
811C0001

And that should be related to DosExit.

That is here:

001B E8 00 00 00 00 call DosExit

and offset 1C needs to be zapped.

So this:

811C0001

is probably split into x'81' having some meaning,
then x'1c00' = 0x001c being an offset, and x'01'
being some section number.

So I'll try to match that to some documentation.

Ok, 2.12 Fixup Record Table appears to have the answer.

And for starters, I'm out by one.

I need to start with:

08811C0001

although I have no idea then what the trailing x'08'
would be before DOSCALLS.

So the x'08' means:

08h = 32-bit Self-relative offset fixup (32-bits).

whatever that means.

Maybe it means that the call is a relative call, so
you need to get the address of DosExit and then
put in a value relative to that.

The x'81' is then broken into:

80h = 8-bit Ordinal Flag.
When set, the ordinal number is 8-bits, otherwise it is 16-bits.

which makes sense, and:

01h = Imported reference by ordinal.

which also makes sense.

1C00 - my offset - matches this:

SRCOFF = DW

which leaves:

01

before the ordinal number.

I can't match that to anything currently.

I'll have another attempt when I'm sober.


BFN. Paul.
Paul Edwards
2024-02-21 02:51:32 UTC
Permalink
Post by Paul Edwards
So I'll try to match that to some documentation.
Ok, 2.12 Fixup Record Table appears to have the answer.
08h = 32-bit Self-relative offset fixup (32-bits).
whatever that means.
80h = 8-bit Ordinal Flag.
When set, the ordinal number is 8-bits, otherwise it is 16-bits.
01h = Imported reference by ordinal.
which also makes sense.
SRCOFF = DW
before the ordinal number.
do you have the LX documentation?
Yes, I quoted some of the IBM documentation above.

BFN. Paul.
Paul Edwards
2024-02-23 22:14:00 UTC
Permalink
Post by Paul Edwards
0001C0 EA08444F 5343414C 4C530000 83EC0489 ..DOSCALLS......
although I have no idea then what the trailing x'08'
would be before DOSCALLS.
Ok, I know what it is now. It's a 1-byte length
field of the name "DOSCALLS", which, as per
2.13 Import Module Name Table, is not NUL-terminated.

So those two extra x'00' before the actual code
starts at x'1cc' are likely padding bytes.

BFN. Paul.
Paul Edwards
2024-02-24 03:38:54 UTC
Permalink
Post by Paul Edwards
Post by Paul Edwards
0001C0 EA08444F 5343414C 4C530000 83EC0489 ..DOSCALLS......
although I have no idea then what the trailing x'08'
would be before DOSCALLS.
Ok, I know what it is now. It's a 1-byte length
field of the name "DOSCALLS", which, as per
2.13 Import Module Name Table, is not NUL-terminated.
So those two extra x'00' before the actual code
starts at x'1cc' are likely padding bytes.
And that offset 1c1 where the import module name table
starts, can be found, is supposedly at 70H in the header.

Which puts it at F0:

000080 4C580000 00000000 02000100 00000000 LX..............
000090 00020000 02000000 01000000 00000000 ................
0000A0 03000000 00080000 00100000 00000000 ................
0000B0 23000000 00000000 64000000 00000000 #.......d.......
0000C0 C4000000 03000000 0C010000 00000000 ................
0000D0 1C010000 00000000 1C010000 27010000 ............'...
0000E0 00000000 00000000 28010000 34010000 ........(...4...
0000F0 41010000 01000000 4A010000 00000000 A.......J.......
000100 CC010000 00000000 00000000 00000000 ................
000110 00000000 02000000 00000000 00000000 ................
000120 00000000 00000000 00000000 00080000 ................

where we can see 0141. I see an interesting 01CC
(ie where my code is) on the next line, which would
be "data pages offset". Not sure what that is.

So back to the 141. I know I need to get to 1C1.
So I'm off by x'80'. So I just need to subtract
the DOS header - cool - the x'141' is the offset
from the LX header.

So trying the x'1cc' again - maybe the data is
loaded at an offset 0x80 from the code, and so
that's the final spot for it? And perhaps the
LX header is expected to be loaded into memory too.

That would require all the other tables to be loaded
too, which would be odd - especially all the relocation
entries which don't serve any purpose once the application
is loaded.

You can always adjust for that though.

Maybe it is set up for a lazy load, and just the data
needs to be memmove'ed to the proper offset if you do
that. Seems odd to be moving it just that small amount
though. ie after accounting for the code, maybe moving
it x'60' bytes. Doesn't sound like a very sensible
alignment.

I see 011c mentioned a couple of times earlier. So
that (likely) corresponds to 019c in the hexdump.
The program name? Someone is interested in that?
Ok, so:

0000D0 1C010000 00000000 1C010000

subtract 80 and you get 50 and 58.

"resource table offset" and "resident
name table offset"? Maybe these are just
dummy pointers, and there is a length
specifier to say the table is empty.

Ok, header x'54' is indeed the number of entries,
and so that (D4 above) is indeed 0, so it's
basically non-existent.

So only the "resident name table offset" is real
perhaps - I don't see a number of entries for it,
so I don't know what that might be.

RESIDENT NAME TBL OFF = DD Resident Name Table offset.
This offset is relative to the beginning of the linear EXE header

Smells correct.

And yes ...

The resident and non-resident name tables define the ASCII names and
ordinal numbers for exported entries
in the module. In addition the first entry in the resident name table
contains the module name.

... the module name is a special first entry.

And indeed, there is no number of entries because:

LEN = DB String Length.
This defines the length of the string in bytes. A zero length indicates
there are no more entries in
table.

000130 00000000 00000000 00000000 00000000 ................
000140 00000000 28000000 00000100 05200000 ....(........ ..
000150 01000000 01000000 00000000 0C000000 ................
000160 00000200 03200000 02000000 01000000 ..... ..........
000170 00000000 00080000 00000300 03200000 ............. ..
000180 03000000 00000000 00000000 00000000 ................
000190 27000000 27000000 0A000000 076F7332 '...'........os2
0001A0 74657374 00000000 00000000 0D000000 test............
0001B0 0D000000 08011000 011A0108 811C0001 ................
0001C0 EA08444F 5343414C 4C530000 83EC0489 ..DOSCALLS......
0001D0 E0506A09 68000002 006A01E8 00000000 .Pj.h....j......
0001E0 83C4106A 006A01E8 00000000 83C40883 ...j.j..........
0001F0 C404C368 690D0A61 62630D0A 00 ...hi..abc...

Still more stuff to be fleshed out.

BFN. Paul.
Paul Edwards
2024-02-24 07:24:25 UTC
Permalink
On 20/02/24 16:10, Paul Edwards wrote:

Ok, another crack at that data.
Post by Paul Edwards
0008 68 00 00 00 00 push offset L$1
This is the push of the data.
Post by Paul Edwards
Segment: CONST DWORD USE32 0000000A bytes
0000 68 69 0D 0A 61 62 63 0D 0A 00 hi..abc...
0001C0 EA08444F 5343414C 4C530000 83EC0489 ..DOSCALLS......
0001D0 E0506A09 68000002 006A01E8 00000000 .Pj.h....j......
68 and a value of 00020000

That looks like a common starting point for data -
2 * 64k. And code is typically placed at 1 * 64k.

Unless relocation information is available and
applied, in which case you can correct all these
values.

Which is a good point - I don't use virtual memory
in PDOS/386 so that address can't possibly stand,
unless it is relative. I doubt that it is a relative
address, but I'm not sufficiently familiar with
x86 opcodes.

Regardless, now would be a good time to find out if
I can find a relocation fixup pointing to that address,
ie offset 9 in the code section.

I don't see such a thing:

D:\devel\pdos_minimal\xxx>hexdump os2test.exe | grep 09
000040 0E1FB409 BA1000CD 21B001B4 4CCD2100 ........!...L.!.
000090 00020000 02000000 01000000 00000000 ................
0001D0 E0506A09 68000002 006A01E8 00000000 .Pj.h....j......

D:\devel\pdos_minimal\xxx>


Maybe the reference is to 1cc+9 or more likely, 14c+9 = 155

D:\devel\pdos_minimal\xxx>hexdump os2test.exe | grep 55

D:\devel\pdos_minimal\xxx>

Nope.

Let me see if there is a wlink option to add relocatable
information.

Nothing stands out with "wlink /?".

Windows executables built by Watcom are (now - just
fixed unrelated issues) working fine under PDOS/386
which means they should contain relocations.

Let me see if the documentation gives any hints.

Offsets 30h, 68h and 6ch are the most likely.
Post by Paul Edwards
0000B0 23000000 00000000 64000000 00000000 #.......d.......
23 as a size is an odd number.
Post by Paul Edwards
0000E0 00000000 00000000 28010000 34010000 ........(...4...
0128 as an offset to something, 0134 as an
offset to something else. So what's there?
Post by Paul Edwards
000190 27000000 27000000 0A000000 076F7332 '...'........os2
0001A0 74657374 00000000 00000000 0D000000 test............
0001B0 0D000000 08011000 011A0108 811C0001 ................
So 1A8 (128 + 80) is 0 and a D and another D.

And 1B4 (134 + 80) is looking like that thing
I documented before with the external names.

I hope OS/2 isn't a system that relies on
virtual memory and doesn't have relocation
capability.
Post by Paul Edwards
I can see that first at 0x1b9.
0001B0 0D000000 08011000 011A0108 811C0001 ................
And indeed - 1B4 is where a DLL fixup is being done.

Fixing up DLL entries is a different concept to
providing relocation information for a module
that has been relocated.

So first thing I need to know is - do OS/2 modules
have a concept of relocation information?

MSDOS 16-bit had that.

From memory NE executables had it too.

Yes, here is some of the NE code I wrote:

fread(numreloc, sizeof numreloc, 1, fp);

nr = numreloc[0] | (numreloc[1] << 8);

/* printf("nr is %d\n", nr); */
for (x = 0; x < nr; x++)
{
/* 2 is segment, 3 is a far address,
5 is an offset */
fread(reloc, sizeof reloc, 1, fp);
if ((reloc[0] != 2)

Is it known by a different name in LX, or is it
non-existent?

This looks interesting:

2.12.5 - Internal Chaining Fixups

Internal chaining fixups are 32-bit offset fixups (source type 07h) to
internal references (target types 00h and
03h) where the first fixup in the chain is a record in the Fixup Record
Table and the remaining fixups are
located in the page referenced by the first fixup rather than as records
in the Fixup Record Table

Looks exactly what I need.

NE is a chain too.

So - where are these?

And it seems to be associated with this:

2.12.4 Internal Entry Table Fixup Record

And probably this:

2.12.1 - Internal Fixup Record

And:

2.12 - Fixup Record Table

which has something I saw earlier:

07h = 32-bit Offset fixup (32-bits).

So - what points to this "Fixup Record Table"?

That's the x'6c' - where I was before.

And as before - I'm missing something that would
allow me to zap that offset.

Either I'm missing relocations in the executable,
or I'm incorrectly thinking that a fixup is needed
in this situation (ie maybe it is relative and the
distance to the data section needs to remain fixed),
or I'm missing some other concept of some sort.

What are all these other numbers anyway?
Post by Paul Edwards
000130 00000000 00000000 00000000 00000000 ................
000140 00000000 28000000 00000100 05200000 ....(........ ..
000150 01000000 01000000 00000000 0C000000 ................
000160 00000200 03200000 02000000 01000000 ..... ..........
000170 00000000 00080000 00000300 03200000 ............. ..
000180 03000000 00000000 00000000 00000000 ................
000190 27000000 27000000 0A000000 076F7332 '...'........os2
Can I see something in the header that points
to them? That would be offset x'B0' to x'120'.
Post by Paul Edwards
000080 4C580000 00000000 02000100 00000000 LX..............
000090 00020000 02000000 01000000 00000000 ................
0000A0 03000000 00080000 00100000 00000000 ................
0000B0 23000000 00000000 64000000 00000000 #.......d.......
0000C0 C4000000 03000000 0C010000 00000000 ................
0000D0 1C010000 00000000 1C010000 27010000 ............'...
0000E0 00000000 00000000 28010000 34010000 ........(...4...
0000F0 41010000 01000000 4A010000 00000000 A.......J.......
000100 CC010000 00000000 00000000 00000000 ................
000110 00000000 02000000 00000000 00000000 ................
000120 00000000 00000000 00000000 00080000 ................
I see a x'00c4' in that range. Nothing else.
That's at C0 = 40, ie object table offset.

number of objects being 3. Not sure what that
3 represents. But let's go to offset C4, ie
144 in the hexdump.
Post by Paul Edwards
000130 00000000 00000000 00000000 00000000 ................
000140 00000000 28000000 00000100 05200000 ....(........ ..
000150 01000000 01000000 00000000 0C000000 ................
000160 00000200 03200000 02000000 01000000 ..... ..........
000170 00000000 00080000 00000300 03200000 ............. ..
000180 03000000 00000000 00000000 00000000 ................
000190 27000000 27000000 0A000000 076F7332 '...'........os2
Before that is all NULs, so could be padding.

And there should be 3 "objects", whatever that is.

Let's see if I can match it to some documentation.
First bit of data is x'28'.

Looks like each entry is x'18' bytes.

And the x'28' is a length - so presumably the length
of the code. That's very close to correct. It should
be x'27' - unless it is size in memory.

And base address is 010000 - reasonable.

Next we have a length of x'c', and a base address of
020000. I'm only expecting 10 bytes of data - so again,
this could be the size when placed in memory, rounded
to a 4-byte boundary.

All very reasonable. What on earth could the 3rd one be?

A length of x'800' - gotta be the stack. Located at x'30000'.
Post by Paul Edwards
000190 27000000 27000000 0A000000 076F7332 '...'........os2
The x'27' is the actual length of the code. Not
sure why it would be listed twice. And x'0a' is
the actual length of the data. That is likely
referenced earlier. Maybe page sizes?

Oh, I missed a bit.
Post by Paul Edwards
000180 00000000 ................
000190 27000000 27000000 0A000000 076F7332 '...'........os2
So offset 0, length 27, and flags.

Then offset 27, length A, and flags.

That is "2.5 Object Page Table".

And that's everything I think. I'm missing the
"internal fixups". Not sure if I missed a wlink
option or something.

BFN. Paul.
Paul Edwards
2024-02-24 07:46:28 UTC
Permalink
Post by Paul Edwards
0000E0 00000000 00000000 28010000 34010000 ........(...4...
0128 as an offset to something, 0134 as an
offset to something else. So what's there?
Post by Paul Edwards
000190 27000000 27000000 0A000000 076F7332 '...'........os2
0001A0 74657374 00000000 00000000 0D000000 test............
0001B0 0D000000 08011000 011A0108 811C0001 ................
0001C0 EA08444F 5343414C 4C530000 83EC0489 ..DOSCALLS......
So 1A8 (128 + 80) is 0 and a D and another D.
I didn't document this.

So x'128' as seen at offset E8 = 68 is
"fixup page table offset", which brings us to:

2.11 Fixup Page Table

There were 2 object page table things, so we are
expecting an extra one, ie 3 fixup entries, with
the last being a pointer to the end. And given
that the last two entries are 'D', it means that
the second object code had no fixups (ie data
doesn't point to other data or code), so it just
points to the end, as a dummy.

And what exactly is the x'd' anyway?

That will be from x'1b4' in the hexdump (ie the DLL
fixups) to just before the "08444F 5343414C 4C53"
(DOSCALLS) at x'1c1'.

Still doesn't get me the required internal fixups.

BFN. Paul.
Paul Edwards
2024-02-24 08:30:02 UTC
Permalink
Post by Paul Edwards
Still doesn't get me the required internal fixups.
Got it!

https://open-watcom.github.io/open-watcom-v2-wikidocs/lguide.html

The INTERNALRELOCS Option

Formats: OS/2

The "INTERNALRELOCS" option is used with LX format executables under
32-bit OS/2. By default, OS/2 executables do not contain internal
relocation information and OS/2 Dynamic Link Libraries do contain
internal relocation information. This option causes the Open Watcom
Linker to include internal relocation information in OS/2 LX format
executables.

The format of the "INTERNALRELOCS" option (short form "INT") is as follows.


OPTION INTERNALRELOCS



Not documented here:

D:\devel\pdos_minimal\xxx>wlink /? | grep -i int
export ::=
entryname["."ordinal]["="internal]["RESident"]["PRIVATE"][iopl]
export ::= entryname["."ordinal]["="internal]["RESident"]["PRIVATE"]
export ::= entryname["."ordinal]["="internal]["PRIVATE"]

D:\devel\pdos_minimal\xxx>


But it works:

wlink File os2test.obj,os2test2.obj Name os2test.exe Form os2 flat
PMCompatible Library os2.lib Option
quiet,start=_os2test,stub=needpdos.exe,internalrelocs


And my executable is 8 bytes bigger:

D:\devel\pdos_minimal\xxx>hexdump os2test.exe
000000 4D5A7C00 01000000 04004100 FFFF0400 MZ|.......A.....
000010 00040000 00000000 40000000 00000000 ***@.......
000020 00000000 00000000 00000000 00000000 ................
000030 00000000 00000000 00000000 80000000 ................
000040 0E1FB409 BA1000CD 21B001B4 4CCD2100 ........!...L.!.
000050 54686973 2070726F 6772616D 206E6565 This program nee
000060 6473204F 532F3220 322E3020 6F722065 ds OS/2 2.0 or e
000070 71756976 616C656E 740D0A24 00000000 quivalent..$....
000080 4C580000 00000000 02000100 00000000 LX..............
000090 10020000 02000000 01000000 00000000 ................
0000A0 03000000 00080000 00100000 00000000 ................
0000B0 2A000000 00000000 64000000 00000000 *.......d.......
0000C0 C4000000 03000000 0C010000 00000000 ................
0000D0 1C010000 00000000 1C010000 27010000 ............'...
0000E0 00000000 00000000 28010000 34010000 ........(...4...
0000F0 48010000 01000000 51010000 00000000 H.......Q.......
000100 D4010000 00000000 00000000 00000000 ................
000110 00000000 02000000 00000000 00000000 ................
000120 00000000 00000000 00000000 00080000 ................
000130 00000000 00000000 00000000 00000000 ................
000140 00000000 28000000 00000100 05200000 ....(........ ..
000150 01000000 01000000 00000000 0C000000 ................
000160 00000200 03200000 02000000 01000000 ..... ..........
000170 00000000 00080000 00000300 03200000 ............. ..
000180 03000000 00000000 00000000 00000000 ................
000190 27000000 27000000 0A000000 076F7332 '...'........os2
0001A0 74657374 00000000 00000000 14000000 test............
0001B0 14000000 08011000 011A0108 811C0001 ................
0001C0 EA070009 00020000 08444F53 43414C4C .........DOSCALL
0001D0 53000000 83EC0489 E0506A09 68000002 S........Pj.h...
0001E0 006A01E8 00000000 83C4106A 006A01E8 .j.........j.j..
0001F0 00000000 83C40883 C404C368 690D0A61 ...........hi..a
000200 62630D0A 00 bc...


And now I can see both the x'07' and the x'09' here:

0001C0 EA070009 00020000 08444F53 43414C4C .........DOSCALL

So that reads:

07h = 32-bit Offset fixup (32-bits).

00h = Internal reference.

0009 = SRCOFF = DW

02 = object index (apparently data)

0000 = offset in the data - my string starts at offset 0
so you don't get to see a nice value there.

BFN. Paul.
Paul Edwards
2024-02-24 15:30:59 UTC
Permalink
I modified my test a little bit.

#include <os2.h>

void os2test(void)
{
ULONG written;

DosWrite(1, "hi\r\n", 4, &written);
DosWrite(1, "abc\r\n", 5, &written);
DosExit(1, 0);
}


.386

.model flat, c

.stack 800h

end



D:\devel\pdos_minimal\xxx>type doit.bat
wcl386 -ecc -D__OS2__ -D__32BIT__ -bt=os2 -l=os2v2 -wx -c -I. -I..\src
-I..\pdpclib -y -fpi87 -s
-zq -3s -zm -zl -oneatblr os2test.c
wasm -zq os2test2.asm
wlink File os2test.obj,os2test2.obj Name os2test.exe Form os2 flat
PMCompatible Library os2.lib
Option quiet,start=_os2test,stub=needpdos.exe,internalrelocs




D:\devel\pdos_minimal\xxx>hexdump os2test.exe
000000 4D5A7C00 01000000 04004100 FFFF0400 MZ|.......A.....
000010 00040000 00000000 40000000 00000000 ***@.......
000020 00000000 00000000 00000000 00000000 ................
000030 00000000 00000000 00000000 80000000 ................
000040 0E1FB409 BA1000CD 21B001B4 4CCD2100 ........!...L.!.
000050 54686973 2070726F 6772616D 206E6565 This program nee
000060 6473204F 532F3220 322E3020 6F722065 ds OS/2 2.0 or e
000070 71756976 616C656E 740D0A24 00000000 quivalent..$....
000080 4C580000 00000000 02000100 00000000 LX..............
000090 10020000 02000000 01000000 00000000 ................
0000A0 03000000 00080000 00100000 00000000 ................
0000B0 38000000 00000000 64000000 00000000 8.......d.......
0000C0 C4000000 03000000 0C010000 00000000 ................
0000D0 1C010000 00000000 1C010000 27010000 ............'...
0000E0 00000000 00000000 28010000 34010000 ........(...4...
0000F0 56010000 01000000 5F010000 00000000 V......._.......
000100 E0010000 00000000 00000000 00000000 ................
000110 00000000 02000000 00000000 00000000 ................
000120 00000000 00000000 00000000 00080000 ................
000130 00000000 00000000 00000000 00000000 ................
000140 00000000 3C000000 00000100 05200000 ....<........ ..
000150 01000000 01000000 00000000 0C000000 ................
000160 00000200 03200000 02000000 01000000 ..... ..........
000170 00000000 00080000 00000300 03200000 ............. ..
000180 03000000 00000000 00000000 00000000 ................
000190 3B000000 3B000000 0B000000 076F7332 ;...;........os2
0001A0 74657374 00000000 00000000 22000000 test........"...
0001B0 22000000 08011000 011A0108 01240001 "............$..
0001C0 1A010881 300001EA 07000900 02000007 ....0...........
0001D0 001D0002 05000844 4F534341 4C4C5300 .......DOSCALLS.
0001E0 83EC0489 E0506A04 68000002 006A01E8 .....Pj.h....j..
0001F0 00000000 83C41089 E0506A05 68050002 .........Pj.h...
000200 006A01E8 00000000 83C4106A 006A01E8 .j.........j.j..
000210 00000000 83C40883 C404C368 690D0A00 ...........hi...
000220 6162630D 0A00 abc...

D:\devel\pdos_minimal\xxx>


And the pdos.zip at http://pdos.org is capable of
loading it.

It is strictly part of proof of concept - I have
hardcoded some values and made other assumptions
just to prove that it works at all, and to allow
pdld development to proceed.

BFN. Paul.
Paul Edwards
2024-02-24 15:28:28 UTC
Permalink
I modified my test a little bit.

#include <os2.h>

void os2test(void)
{
ULONG written;

DosWrite(1, "hi\r\n", 4, &written);
DosWrite(1, "abc\r\n", 5, &written);
DosExit(1, 0);
}


.386

.model flat, c

.stack 800h

end



D:\devel\pdos_minimal\xxx>type doit.bat
wcl386 -ecc -D__OS2__ -D__32BIT__ -bt=os2 -l=os2v2 -wx -c -I. -I..\src
-I..\pdpclib -y -fpi87 -s
-zq -3s -zm -zl -oneatblr os2test.c
wasm -zq os2test2.asm
wlink File os2test.obj,os2test2.obj Name os2test.exe Form os2 flat
PMCompatible Library os2.lib
Option quiet,start=_os2test,stub=needpdos.exe,internalrelocs




D:\devel\pdos_minimal\xxx>hexdump os2test.exe
000000 4D5A7C00 01000000 04004100 FFFF0400 MZ|.......A.....
000010 00040000 00000000 40000000 00000000 ***@.......
000020 00000000 00000000 00000000 00000000 ................
000030 00000000 00000000 00000000 80000000 ................
000040 0E1FB409 BA1000CD 21B001B4 4CCD2100 ........!...L.!.
000050 54686973 2070726F 6772616D 206E6565 This program nee
000060 6473204F 532F3220 322E3020 6F722065 ds OS/2 2.0 or e
000070 71756976 616C656E 740D0A24 00000000 quivalent..$....
000080 4C580000 00000000 02000100 00000000 LX..............
000090 10020000 02000000 01000000 00000000 ................
0000A0 03000000 00080000 00100000 00000000 ................
0000B0 38000000 00000000 64000000 00000000 8.......d.......
0000C0 C4000000 03000000 0C010000 00000000 ................
0000D0 1C010000 00000000 1C010000 27010000 ............'...
0000E0 00000000 00000000 28010000 34010000 ........(...4...
0000F0 56010000 01000000 5F010000 00000000 V......._.......
000100 E0010000 00000000 00000000 00000000 ................
000110 00000000 02000000 00000000 00000000 ................
000120 00000000 00000000 00000000 00080000 ................
000130 00000000 00000000 00000000 00000000 ................
000140 00000000 3C000000 00000100 05200000 ....<........ ..
000150 01000000 01000000 00000000 0C000000 ................
000160 00000200 03200000 02000000 01000000 ..... ..........
000170 00000000 00080000 00000300 03200000 ............. ..
000180 03000000 00000000 00000000 00000000 ................
000190 3B000000 3B000000 0B000000 076F7332 ;...;........os2
0001A0 74657374 00000000 00000000 22000000 test........"...
0001B0 22000000 08011000 011A0108 01240001 "............$..
0001C0 1A010881 300001EA 07000900 02000007 ....0...........
0001D0 001D0002 05000844 4F534341 4C4C5300 .......DOSCALLS.
0001E0 83EC0489 E0506A04 68000002 006A01E8 .....Pj.h....j..
0001F0 00000000 83C41089 E0506A05 68050002 .........Pj.h...
000200 006A01E8 00000000 83C4106A 006A01E8 .j.........j.j..
000210 00000000 83C40883 C404C368 690D0A00 ...........hi...
000220 6162630D 0A00 abc...

D:\devel\pdos_minimal\xxx>


And the pdos.zip at http://pdos.org is capable of
loading it.

It is strictly part of proof of concept - I have
hardcoded some values and made other assumptions
just to prove that it works at all, and to allow
pdld development to proceed.

BFN. Paul.
Paul Edwards
2024-02-25 22:56:04 UTC
Permalink
Post by Paul Edwards
It is strictly part of proof of concept - I have
hardcoded some values and made other assumptions
just to prove that it works at all, and to allow
pdld development to proceed.
The author of pdld has started adding support for LX:

https://sourceforge.net/p/pdos/gitcode/ci/master/tree/pdld/src/lx.c

and also removed some hardcoding from the loader:

https://sourceforge.net/p/pdos/gitcode/ci/master/tree/bios/exeload.c

but it's still not usable for anything other than
limited testing.

With a bit of luck, that will change within 24 hours.

BFN. Paul.
Peter Flass
2024-02-27 22:41:26 UTC
Permalink
Post by Paul Edwards
Post by Paul Edwards
It is strictly part of proof of concept - I have
hardcoded some values and made other assumptions
just to prove that it works at all, and to allow
pdld development to proceed.
https://sourceforge.net/p/pdos/gitcode/ci/master/tree/pdld/src/lx.c
https://sourceforge.net/p/pdos/gitcode/ci/master/tree/bios/exeload.c
but it's still not usable for anything other than
limited testing.
With a bit of luck, that will change within 24 hours.
BFN. Paul.
Thank you, this is very interesting. I’m wondering if it’s possible to run
OS/2 executables on Linux, with suitable work.
--
Pete
Paul Edwards
2024-02-28 04:50:24 UTC
Permalink
Post by Peter Flass
Post by Paul Edwards
https://sourceforge.net/p/pdos/gitcode/ci/master/tree/bios/exeload.c
but it's still not usable for anything other than
limited testing.
With a bit of luck, that will change within 24 hours.
OS/2 didn't move overnight (ie Slovakia daytime)
because focus was switched to supporting Linux ELF,
which has made gigantic strides and I should be
back to that later today.
Post by Peter Flass
Thank you, this is very interesting. I’m wondering if it’s possible to run
OS/2 executables on Linux, with suitable work.
Yes, for a similar reason that certain Win32 programs
can be run under Linux, a similar (but different
because of the intercept at doscalls.dll rather than
msvcrt.dll) technique proved to be possible for OS/2,
and if you go to http://pdos.org and search for the
second occurrence of OS/2, you will see uc386l.zip
which contains a standalone (no dependency even on
libc) 50k bios.exe Linux ELF binary which is capable
of running certain Win32 executables (some provided
in the distribution), and very limited (like the one
I posted here as a hexdump) OS/2 executables.

There is no barrier I know of for this to be further
fleshed out to support any C90 application - and with
limited increase to the 50k size.

The main barrier is the pdld author is grappling with
some OS/2 executable issues and I don't want to outrun
him (and he is the safer pair of hands if he chooses
to modify exeload in tandem).

He seemed annoyed that he couldn't access doscalls
functions by name instead of ordinal and asked me if
there was a reason for that. Anyone know?

He also had issues with "object" (LX) vs "section" (PE)
but it is unclear to me whether they are conceptually
different.

BFN. Paul.
Dave Yeo
2024-02-28 22:04:19 UTC
Permalink
Post by Paul Edwards
He seemed annoyed that he couldn't access doscalls
functions by name instead of ordinal and asked me if
there was a reason for that. Anyone know?
Usual use of ordinals was to save memory. Important when running with
4MB's of ram like when I started.
Usually link to doscall1.dll, rather then directly to the kernel, where
doscall1.dll does seem to export by name. Just generated a DEF file from
the SMP version of doscall1.dll,
;
; doscall1.def (created by emximp)
;
LIBRARY doscall1 INITINSTANCE TERMGLOBAL
EXPORTS
"DOSTIMERASYNC"
"DOSTIMERSTART"
"DOSTIMERSTOP"
"DOSREAD"
"DOSWRITE"
"DOSERRCLASS"
"DOSSEMREQUEST"
"DOSSEMCLEAR"
"DOSSEMWAIT"
"DOSSEMSET"
"DOSEXECPGM"
"DOSSUBSET"
"DOSSUBALLOC"
"DOSSUBFREE"
"DOSREADASYNC"
"DOSWRITEASYNC"
"DOSSEARCHPATH"
"DOSSCANENV"
"DOSSETCP"
"DOSCALLBACK"
"DOSFSRAMSEMREQUEST"
"DOSFSRAMSEMCLEAR"
"DOSQAPPTYPE"
"DOSSETPROCCP"
"DOSLOGREGISTER"
"DOSLOGREAD"
"DOSCOPY"
"DOSPMSEMWAIT"
"DOSPMMUXSEMWAIT"
"DOSSGQUERYTOPMOST"
"DOSSETEXTLIBPATH"
"DOSQUERYEXTLIBPATH"

Dave
Paul Edwards
2024-02-28 22:56:17 UTC
Permalink
Hi Dave.

Thanks for the explanation.

That seems to be missing DOSOPEN as well as others.

Note that I received an enhancement to exeload.c
overnight which should allow for much more robust
loading of executables and I'll see what it is
capable of later (source has been committed
already though, and makebios.lnp has the
instructions).

Also the pdld author's concerns appear to have been
allayed.

I will add the missing function names to exeload.c,
which is trivial, and see what happens when I load
pdptest.exe built by Watcom.

I would still need to flesh out my doscalls.c, but
that shouldn't be an issue to get pdptest.exe to run.

BFN. Paul.
Post by Dave Yeo
Post by Paul Edwards
He seemed annoyed that he couldn't access doscalls
functions by name instead of ordinal and asked me if
there was a reason for that. Anyone know?
Usual use of ordinals was to save memory. Important when running with
4MB's of ram like when I started.
Usually link to doscall1.dll, rather then directly to the kernel, where
doscall1.dll does seem to export by name. Just generated a DEF file from
the SMP version of doscall1.dll,
;
; doscall1.def (created by emximp)
;
LIBRARY doscall1 INITINSTANCE TERMGLOBAL
EXPORTS
"DOSTIMERASYNC"
"DOSTIMERSTART"
"DOSTIMERSTOP"
"DOSREAD"
"DOSWRITE"
"DOSERRCLASS"
"DOSSEMREQUEST"
"DOSSEMCLEAR"
"DOSSEMWAIT"
"DOSSEMSET"
"DOSEXECPGM"
"DOSSUBSET"
"DOSSUBALLOC"
"DOSSUBFREE"
"DOSREADASYNC"
"DOSWRITEASYNC"
"DOSSEARCHPATH"
"DOSSCANENV"
"DOSSETCP"
"DOSCALLBACK"
"DOSFSRAMSEMREQUEST"
"DOSFSRAMSEMCLEAR"
"DOSQAPPTYPE"
"DOSSETPROCCP"
"DOSLOGREGISTER"
"DOSLOGREAD"
"DOSCOPY"
"DOSPMSEMWAIT"
"DOSPMMUXSEMWAIT"
"DOSSGQUERYTOPMOST"
"DOSSETEXTLIBPATH"
"DOSQUERYEXTLIBPATH"
Dave
Dave Yeo
2024-02-29 03:11:58 UTC
Permalink
Post by Paul Edwards
Hi Dave.
Thanks for the explanation.
That seems to be missing DOSOPEN as well as others.
Yes it does. Wouldn't be surprised if emximp.exe has problems with
doscall1.dll, perhaps failing with the 16bit exports.
Dave
Paul Edwards
2024-02-29 03:46:10 UTC
Permalink
Post by Dave Yeo
Post by Paul Edwards
Hi Dave.
Thanks for the explanation.
That seems to be missing DOSOPEN as well as others.
Yes it does. Wouldn't be surprised if emximp.exe has problems with
doscall1.dll, perhaps failing with the 16bit exports.
The string doesn't appear to be in the DLL at
all, unlike e.g. DosWrite().

C:\w2kshare>hexdump DOSCALL1.DLL | grep -i open

C:\w2kshare>hexdump DOSCALL1.DLL | grep -i doso

C:\w2kshare>hexdump DOSCALL1.DLL | grep -i doswr
000410 4F535245 41448900 08444F53 57524954 OSREAD...DOSWRIT

C:\w2kshare>


BFN. Paul.
Dave Yeo
2024-02-29 07:02:26 UTC
Permalink
Post by Paul Edwards
Post by Dave Yeo
Post by Paul Edwards
Hi Dave.
Thanks for the explanation.
That seems to be missing DOSOPEN as well as others.
Yes it does. Wouldn't be surprised if emximp.exe has problems with
doscall1.dll, perhaps failing with the 16bit exports.
The string doesn't appear to be in the DLL at
all, unlike e.g. DosWrite().
os2386.lib and os2286.lib seem to point it as being in doscalls. I also
see it in the kernel's sym file, on the ISO under \os2\image\debug are
the half strict kernels and corresponding sym etc files. No experience
with them.
It's all learning for me with some stuff I thought I knew being wrong it
seems.
Dave
Paul Edwards
2024-02-29 23:15:14 UTC
Permalink
Post by Dave Yeo
Post by Paul Edwards
Post by Dave Yeo
Post by Paul Edwards
Hi Dave.
Thanks for the explanation.
That seems to be missing DOSOPEN as well as others.
Yes it does. Wouldn't be surprised if emximp.exe has problems with
doscall1.dll, perhaps failing with the 16bit exports.
The string doesn't appear to be in the DLL at
all, unlike e.g. DosWrite().
os2386.lib and os2286.lib seem to point it as being in doscalls.
Sure - that's for linking purposes, where it MUST
get resolved to an ordinal, because the DLL doesn't
have the name information (it seems). Which was the
query.

Regardless, I wasn't able to get to the code yesterday,
but fortunately I received a code change from Slovakia
overnight that should make loading much more robust,
and I will try it out today hopefully and hopefully
fill in the missing functionality to get the pdptest
program (makefile.wat), including parameters, working.

Or find out that I am missing some concept.
Post by Dave Yeo
I also
see it in the kernel's sym file, on the ISO under \os2\image\debug are
the half strict kernels and corresponding sym etc files. No experience
with them.
It's all learning for me with some stuff I thought I knew being wrong it
seems.
Ok. :-)

BFN. Paul.

Dave Yeo
2024-02-28 22:20:02 UTC
Permalink
Post by Paul Edwards
He seemed annoyed that he couldn't access doscalls
functions by name instead of ordinal and asked me if
there was a reason for that. Anyone know?
There's also the RESIDENTNAME key word. From the toolkits documentation,
Tools Reference. Ignore the pwords stuff, doesn't apply to protected
mode as far as I know.
...
Export-Definition Syntax

entryname [=internalname] [@ord[RESIDENTNAME]] [pwords]

<entryname>
The function name as it is known to other modules.

<internalname>
The actual name of the export function as it appears within the module
itself; by default, this name is the same as <entryname>.

<ord>
The function's ordinal position within the module definition table. If
this field is used, the function's entry point can be invoked by name or
by ordinal. Use of ordinal positions is faster and may save space.

RESIDENTNAME
Indicates that the function's name be kept resident in memory at all
times. This keyword is applicable only if <ord> is used. If <ord> is not
used, OS/2 automatically keeps the names of all exported functions
resident in memory by default.

<pwords>
The total size of the function's parameters, as measured in words (bytes
divided by two). This field is required only if the function executes
with I/O privilege. When a function with I/O privilege is called, OS/2
consults <pwords> to determine how many words to copy from the caller's
stack to the I/O-privileged function's stack.

Example

EXPORTS
SampleRead = read2bin @8
StringIn = str1 @4 RESIDENTNAME
CharTest 6

This example defines three export functions:

SampleRead
StringIn
CharTest

The first two functions can be accessed either by their exported names
or by an ordinal number. Note that in the module's own source code,
these functions are actually defined as read2bin and str1, respectively.
The last function runs with I/O privilege and therefore is given with
the total size of the parameters - six words.
...
Dave
Dave Yeo
2024-02-28 21:52:33 UTC
Permalink
Post by Peter Flass
Thank you, this is very interesting. I’m wondering if it’s possible to run
OS/2 executables on Linux, with suitable work.
There's 2ine, needs more work of course. Couple of links,
https://virtuallyfun.com/2018/02/23/2ine-the-os-2-emulator/
https://virtuallyfun.com/2018/06/11/2ine-updated-to-have-preliminary-16-bit-exe-support/
Dave
Peter Flass
2024-02-29 00:43:10 UTC
Permalink
Post by Dave Yeo
Post by Peter Flass
Thank you, this is very interesting. I’m wondering if it’s possible to run
OS/2 executables on Linux, with suitable work.
There's 2ine, needs more work of course. Couple of links,
https://virtuallyfun.com/2018/02/23/2ine-the-os-2-emulator/
https://virtuallyfun.com/2018/06/11/2ine-updated-to-have-preliminary-16-bit-exe-support/
Dave
What a coincidence. I was thinking “twine”.
--
Pete
Loading...