1 <chapter id="implementation">
2 <title>Low-level Implementation</title>
3 <para>Details of Wine's Low-level Implementation...</para>
5 <sect1 id="config-keyboard">
6 <title>Keyboard</title>
9 Wine now needs to know about your keyboard layout. This
10 requirement comes from a need from many apps to have the
11 correct scancodes available, since they read these directly,
12 instead of just taking the characters returned by the X
13 server. This means that Wine now needs to have a mapping from
14 X keys to the scancodes these programs expect.
17 On startup, Wine will try to recognize the active X layout by
18 seeing if it matches any of the defined tables. If it does,
19 everything is alright. If not, you need to define it.
22 To do this, open the file
23 <filename>dlls/x11drv/keyboard.c</filename> and take a look
24 at the existing tables. Make a backup copy of it, especially
28 What you really would need to do, is find out which scancode
29 each key needs to generate. Find it in the
30 <function>main_key_scan</function> table, which looks like
34 static const int main_key_scan[MAIN_LEN] =
36 /* this is my (102-key) keyboard layout, sorry if it doesn't quite match yours */
37 0x29,0x02,0x03,0x04,0x05,0x06,0x07,0x08,0x09,0x0A,0x0B,0x0C,0x0D,
38 0x10,0x11,0x12,0x13,0x14,0x15,0x16,0x17,0x18,0x19,0x1A,0x1B,
39 0x1E,0x1F,0x20,0x21,0x22,0x23,0x24,0x25,0x26,0x27,0x28,0x2B,
40 0x2C,0x2D,0x2E,0x2F,0x30,0x31,0x32,0x33,0x34,0x35,
41 0x56 /* the 102nd key (actually to the right of l-shift) */
45 Next, assign each scancode the characters imprinted on the
46 keycaps. This was done (sort of) for the US 101-key keyboard,
47 which you can find near the top in
48 <filename>keyboard.c</filename>. It also shows that if there
49 is no 102nd key, you can skip that.
52 However, for most international 102-key keyboards, we have
53 done it easy for you. The scancode layout for these already
54 pretty much matches the physical layout in the
55 <function>main_key_scan</function>, so all you need to do is
56 to go through all the keys that generate characters on your
57 main keyboard (except spacebar), and stuff those into an
58 appropriate table. The only exception is that the 102nd key,
59 which is usually to the left of the first key of the last line
60 (usually <keycap>Z</keycap>), must be placed on a separate
61 line after the last line.
64 For example, my Norwegian keyboard looks like this
67 § ! " # ¤ % & / ( ) = ? ` Back-
68 | 1 2@ 3£ 4$ 5 6 7{ 8[ 9] 0} + \´ space
70 Tab Q W E R T Y U I O P Å ^
73 Caps A S D F G H J K L Ø Æ *
76 Sh- > Z X C V B N M ; : _ Shift
79 Ctrl Alt Spacebar AltGr Ctrl
82 Note the 102nd key, which is the <keycap><></keycap> key, to
83 the left of <keycap>Z</keycap>. The character to the right of
84 the main character is the character generated by
85 <keycap>AltGr</keycap>.
88 This keyboard is defined as follows:
91 static const char main_key_NO[MAIN_LEN][4] =
93 "|§","1!","2\"@","3#£","4¤$","5%","6&","7/{","8([","9)]","0=}","+?","\\´",
94 "qQ","wW","eE","rR","tT","yY","uU","iI","oO","pP","åÅ","¨^~",
95 "aA","sS","dD","fF","gG","hH","jJ","kK","lL","øØ","æÆ","'*",
96 "zZ","xX","cC","vV","bB","nN","mM",",;",".:","-_",
101 Except that " and \ needs to be quoted with a backslash, and
102 that the 102nd key is on a separate line, it's pretty
106 After you have written such a table, you need to add it to the
107 <function>main_key_tab[]</function> layout index table. This
112 WORD lang, ansi_codepage, oem_codepage;
113 const char (*key)[MAIN_LEN][4];
117 {MAKELANGID(LANG_NORWEGIAN,SUBLANG_DEFAULT), 1252, 865, &main_key_NO},
121 After you have added your table, recompile Wine and test that
122 it works. If it fails to detect your table, try running
125 wine --debugmsg +key,+keyboard >& key.log
128 and look in the resulting <filename>key.log</filename> file to
129 find the error messages it gives for your layout.
132 Note that the <constant>LANG_*</constant> and
133 <constant>SUBLANG_*</constant> definitions are in
134 <filename>include/winnls.h</filename>, which you might need to
135 know to find out which numbers your language is assigned, and
136 find it in the debugmsg output. The numbers will be
137 <literal>(SUBLANG * 0x400 + LANG)</literal>, so, for example
138 the combination <literal>LANG_NORWEGIAN (0x14)</literal> and
139 <literal>SUBLANG_DEFAULT (0x1)</literal> will be (in hex)
140 <literal>14 + 1*400 = 414</literal>, so since I'm Norwegian, I
141 could look for <literal>0414</literal> in the debugmsg output
142 to find out why my keyboard won't detect.
145 Once it works, submit it to the Wine project. If you use CVS,
146 you will just have to do
149 cvs -z3 diff -u dlls/x11drv/keyboard.c > layout.diff
152 from your main Wine directory, then submit
153 <filename>layout.diff</filename> to
154 <email>wine-patches@winehq.org</email> along with a brief note
158 If you don't use CVS, you need to do
161 diff -u the_backup_file_you_made dlls/x11drv/keyboard.c > layout.diff
164 and submit it as explained above.
167 If you did it right, it will be included in the next Wine
168 release, and all the troublesome programs (especially
169 remote-control programs) and games that use scancodes will
170 be happily using your keyboard layout, and you won't get those
171 annoying fixme messages either.
176 <sect1 id="undoc-func">
177 <title>Undocumented APIs</title>
180 Some background: On the i386 class of machines, stack entries are
181 usually dword (4 bytes) in size, little-endian. The stack grows
182 downward in memory. The stack pointer, maintained in the
183 <literal>esp</literal> register, points to the last valid entry;
184 thus, the operation of pushing a value onto the stack involves
185 decrementing <literal>esp</literal> and then moving the value into
186 the memory pointed to by <literal>esp</literal>
187 (i.e., <literal>push p</literal> in assembly resembles
188 <literal>*(--esp) = p;</literal> in C). Removing (popping)
189 values off the stack is the reverse (i.e., <literal>pop p</literal>
190 corresponds to <literal>p = *(esp++);</literal> in C).
194 In the <literal>stdcall</literal> calling convention, arguments are
195 pushed onto the stack right-to-left. For example, the C call
196 <function>myfunction(40, 20, 70, 30);</function> is expressed in
205 The called function is responsible for removing the arguments
206 off the stack. Thus, before the call to myfunction, the
207 stack would look like:
209 [local variable or temporary]
210 [local variable or temporary]
216 After the call returns, it should look like:
218 [local variable or temporary]
219 esp -> [local variable or temporary]
224 To restore the stack to this state, the called function must know how
225 many arguments to remove (which is the number of arguments it takes).
226 This is a problem if the function is undocumented.
230 One way to attempt to document the number of arguments each function
231 takes is to create a wrapper around that function that detects the
232 stack offset. Essentially, each wrapper assumes that the function will
233 take a large number of arguments. The wrapper copies each of these
234 arguments into its stack, calls the actual function, and then calculates
235 the number of arguments by checking esp before and after the call.
239 The main problem with this scheme is that the function must actually
240 be called from another program. Many of these functions are seldom
241 used. An attempt was made to aggressively query each function in a
242 given library (<filename>ntdll.dll</filename>) by passing 64 arguments,
243 all 0, to each function. Unfortunately, Windows NT quickly goes to a
244 blue screen of death, even if the program is run from a
245 non-administrator account.
249 Another method that has been much more successful is to attempt to
250 figure out how many arguments each function is removing from the
251 stack. This instruction, <literal>ret hhll</literal> (where
252 <symbol>hhll</symbol> is the number of bytes to remove, i.e. the
253 number of arguments times 4), contains the bytes
254 <literal>0xc2 ll hh</literal> in memory. It is a reasonable
255 assumption that few, if any, functions take more than 16 arguments;
256 therefore, simply searching for
257 <literal>hh == 0 && ll < 0x40</literal> starting from the
258 address of a function yields the correct number of arguments most
263 Of course, this is not without errors. <literal>ret 00ll</literal>
264 is not the only instruction that can have the byte sequence
265 <literal>0xc2 ll 0x0</literal>; for example,
266 <literal>push 0x000040c2</literal> has the byte sequence
267 <literal>0x68 0xc2 0x40 0x0 0x0</literal>, which matches
268 the above. Properly, the utility should look for this sequence
269 only on an instruction boundary; unfortunately, finding
270 instruction boundaries on an i386 requires implementing a full
271 disassembler -- quite a daunting task. Besides, the probability
272 of having such a byte sequence that is not the actual return
273 instruction is fairly low.
277 Much more troublesome is the non-linear flow of a function. For
278 example, consider the following two functions:
281 jmp somefunction1_impl
289 In this case, we would incorrectly detect both
290 <function>somefunction1</function> and
291 <function>somefunction2</function> as taking only a single
292 argument, whereas <function>somefunction1</function> really
297 With these limitations in mind, it is possible to implement more stubs
298 in Wine and, eventually, the functions themselves.
302 <sect1 id="accel-impl">
303 <title>Accelerators</title>
306 There are <emphasis>three</emphasis> differently sized
307 accelerator structures exposed to the user:
312 Accelerators in NE resources. This is also the internal
313 layout of the global handle <type>HACCEL</type> (16 and
314 32) in Windows 95 and Wine. Exposed to the user as Win16
315 global handles <type>HACCEL16</type> and
316 <type>HACCEL32</type> by the Win16/Win32 API.
317 These are 5 bytes long, with no padding:
327 Accelerators in PE resources. They are exposed to the user
328 only by direct accessing PE resources.
329 These have a size of 8 bytes:
341 Accelerators in the Win32 API. These are exposed to the
342 user by the <function>CopyAcceleratorTable</function>
343 and <function>CreateAcceleratorTable</function> functions
345 These have a size of 6 bytes:
357 Why two types of accelerators in the Win32 API? We can only
358 guess, but my best bet is that the Win32 resource compiler
359 can/does not handle struct packing. Win32 <type>ACCEL</type>
360 is defined using <function>#pragma(2)</function> for the
361 compiler but without any packing for RC, so it will assume
362 <function>#pragma(4)</function>.
367 <sect1 id="hardware-trace">
368 <title>Doing A Hardware Trace</title>
371 The primary reason to do this is to reverse engineer a
372 hardware device for which you don't have documentation, but
373 can get to work under Wine.
376 This lot is aimed at parallel port devices, and in particular
377 parallel port scanners which are now so cheap they are
378 virtually being given away. The problem is that few
379 manufactures will release any programming information which
380 prevents drivers being written for Sane, and the traditional
381 technique of using DOSemu to produce the traces does not work
382 as the scanners invariably only have drivers for Windows.
385 Presuming that you have compiled and installed wine the first
386 thing to do is is to enable direct hardware access to your
387 parallel port. To do this edit <filename>config</filename>
388 (usually in <filename>~/.wine/</filename>) and in the
389 ports section add the following two lines
392 read=0x378,0x379,0x37a,0x37c,0x77a
393 write=0x378,x379,0x37a,0x37c,0x77a
396 This adds the necessary access required for SPP/PS2/EPP/ECP
397 parallel port on LPT1. You will need to adjust these number
398 accordingly if your parallel port is on LPT2 or LPT0.
401 When starting wine use the following command line, where
402 <literal>XXXX</literal> is the program you need to run in
403 order to access your scanner, and <literal>YYYY</literal> is
404 the file your trace will be stored in:
407 wine -debugmsg +io XXXX 2> >(sed 's/^[^:]*:io:[^ ]* //' > YYYY)
410 You will need large amounts of hard disk space (read hundreds
411 of megabytes if you do a full page scan), and for reasonable
412 performance a really fast processor and lots of RAM.
415 You will need to postprocess the output into a more manageable
416 format, using the <command>shrink</command> program. First
417 you need to compile the source (which is located at the end of
420 cc shrink.c -o shrink
424 Use the <command>shrink</command> program to reduce the
425 physical size of the raw log as follows:
428 cat log | shrink > log2
431 The trace has the basic form of
434 XXXX > YY @ ZZZZ:ZZZZ
437 where <literal>XXXX</literal> is the port in hexidecimal being
438 accessed, <literal>YY</literal> is the data written (or read)
439 from the port, and <literal>ZZZZ:ZZZZ</literal> is the address
440 in memory of the instruction that accessed the port. The
441 direction of the arrow indicates whether the data was written
442 or read from the port.
445 > data was written to the port
446 < data was read from the port
449 My basic tip for interpreting these logs is to pay close
450 attention to the addresses of the IO instructions. Their
451 grouping and sometimes proximity should reveal the presence of
452 subroutines in the driver. By studying the different versions
453 you should be able to work them out. For example consider the
454 following section of trace from my UMAX Astra 600P
457 0x378 > 55 @ 0297:01ec
458 0x37a > 05 @ 0297:01f5
459 0x379 < 8f @ 0297:01fa
460 0x37a > 04 @ 0297:0211
461 0x378 > aa @ 0297:01ec
462 0x37a > 05 @ 0297:01f5
463 0x379 < 8f @ 0297:01fa
464 0x37a > 04 @ 0297:0211
465 0x378 > 00 @ 0297:01ec
466 0x37a > 05 @ 0297:01f5
467 0x379 < 8f @ 0297:01fa
468 0x37a > 04 @ 0297:0211
469 0x378 > 00 @ 0297:01ec
470 0x37a > 05 @ 0297:01f5
471 0x379 < 8f @ 0297:01fa
472 0x37a > 04 @ 0297:0211
473 0x378 > 00 @ 0297:01ec
474 0x37a > 05 @ 0297:01f5
475 0x379 < 8f @ 0297:01fa
476 0x37a > 04 @ 0297:0211
477 0x378 > 00 @ 0297:01ec
478 0x37a > 05 @ 0297:01f5
479 0x379 < 8f @ 0297:01fa
480 0x37a > 04 @ 0297:0211
483 As you can see there is a repeating structure starting at
484 address <literal>0297:01ec</literal> that consists of four io
485 accesses on the parallel port. Looking at it the first io
486 access writes a changing byte to the data port the second
487 always writes the byte <literal>0x05</literal> to the control
488 port, then a value which always seems to
489 <literal>0x8f</literal> is read from the status port at which
490 point a byte <literal>0x04</literal> is written to the control
491 port. By studying this and other sections of the trace we can
492 write a C routine that emulates this, shown below with some
493 macros to make reading/writing on the parallel port easier to
497 #define r_dtr(x) inb(x)
498 #define r_str(x) inb(x+1)
499 #define r_ctr(x) inb(x+2)
500 #define w_dtr(x,y) outb(y, x)
501 #define w_str(x,y) outb(y, x+1)
502 #define w_ctr(x,y) outb(y, x+2)
504 /* Seems to be sending a command byte to the scanner */
505 int udpp_put(int udpp_base, unsigned char command)
509 w_dtr(udpp_base, command);
510 w_ctr(udpp_base, 0x05);
512 for (loop=0; loop < 10; loop++)
513 if ((value = r_str(udpp_base)) & 0x80)
515 w_ctr(udpp_base, 0x04);
519 return (value & 0xf8) | 0x01;
523 For the UMAX Astra 600P only seven such routines exist (well
524 14 really, seven for SPP and seven for EPP). Whether you
525 choose to disassemble the driver at this point to verify the
526 routines is your own choice. If you do, the address from the
527 trace should help in locating them in the disassembly.
530 You will probably then find it useful to write a script/perl/C
531 program to analyse the logfile and decode them futher as this
532 can reveal higher level grouping of the low level routines.
533 For example from the logs from my UMAX Astra 600P when decoded
534 further reveal (this is a small snippet)
559 From this it is easy to see that <varname>put</varname>
560 routine is often grouped together in five successive calls
561 sending information to the scanner. Once these are understood
562 it should be possible to process the logs further to show the
563 higher level routines in an easy to see format. Once the
564 highest level format that you can derive from this process is
565 understood, you then need to produce a series of scans varying
566 only one parameter between them, so you can discover how to
567 set the various parameters for the scanner.
571 The following is the <filename>shrink.c</filename> program:
573 /* Copyright David Campbell <campbell@torque.net> */
574 #include <stdio.h>
575 #include <string.h>
579 char buff[256], lastline[256] = "";
582 while (!feof (stdin))
584 fgets (buff, sizeof (buff), stdin);
585 if (strcmp (buff, lastline))
588 printf ("# Last line repeated %i times #\n", count);
590 strcpy (lastline, buff);
603 <!-- Keep this comment at the end of the file
606 sgml-parent-document:("wine-devel.sgml" "set" "book" "part" "chapter" "")