|  | <chapter id="debugging"> | 
|  | <title>Debugging Wine</title> | 
|  |  | 
|  | <sect1 id="debug-msg"> | 
|  | <title>Debug Messages</title> | 
|  |  | 
|  | <para> | 
|  | written by Dimitrie O. Paun <email>dimi@cs.toronto.edu</email>, 28 Mar 1998 | 
|  | </para> | 
|  | <para> | 
|  | (Extracted from <filename>wine/documentation/debug-msgs</filename>) | 
|  | </para> | 
|  |  | 
|  | <note> | 
|  | <para> | 
|  | The new debugging interface can be considered to be | 
|  | stable, with the exception of the in-memory message | 
|  | construction functions. However, there is still a lot of | 
|  | work to be done to polish things up. To make my life | 
|  | easier, please follow the guidelines described in this | 
|  | document. | 
|  | </para> | 
|  | </note> | 
|  |  | 
|  | <important> | 
|  | <para> | 
|  | Read this document before writing new code. DO NOT USE | 
|  | <function>fprintf</function>  (or | 
|  | <function>printf</function>) to output things. Also, instead | 
|  | of writing  FIXMEs in the source, output a FIXME message if | 
|  | you can. | 
|  | </para> | 
|  | <para> | 
|  | At the end of the document, there is a "Style Guide" for | 
|  | debugging messages. Please read it. | 
|  | </para> | 
|  | </important> | 
|  |  | 
|  | <sect2> | 
|  | <title>Debugging classes</title> | 
|  |  | 
|  | <para> | 
|  | There are 4 types (or classes) of debugging messages: | 
|  | </para> | 
|  | <variablelist> | 
|  | <varlistentry> | 
|  | <term><literal>FIXME</literal></term> | 
|  | <listitem> | 
|  | <para> | 
|  | Messages in this class relate to behavior of Wine that | 
|  | does not correspond to standard Windows behavior and | 
|  | that should be fixed. | 
|  | </para> | 
|  | <para>Examples: stubs, semi-implemented features, etc.</para> | 
|  | </listitem> | 
|  | </varlistentry> | 
|  | <varlistentry> | 
|  | <term><literal>ERR</literal></term> | 
|  | <listitem> | 
|  | <para> | 
|  | Messages in this class relate to serious errors in | 
|  | Wine. This sort of messages are close to asserts -- | 
|  | that is, you should output an error message when the | 
|  | code detects a condition which should not happen. In | 
|  | other words, important things that are not warnings | 
|  | (see below), are errors. | 
|  | </para> | 
|  | <para> | 
|  | Examples: unexpected change in internal state, etc. | 
|  | </para> | 
|  | </listitem> | 
|  | </varlistentry> | 
|  | <varlistentry> | 
|  | <term><literal>WARN</literal></term> | 
|  | <listitem> | 
|  | <para> | 
|  | These are warning messages. You should report a | 
|  | warning when something unwanted happen but the | 
|  | function behaves properly. That is, output a warning | 
|  | when you encounter something unexpected (ex: could not | 
|  | open a file) but the function deals correctly with the | 
|  | situation (that is, according to the docs). If you do | 
|  | not deal correctly with it, output a fixme. | 
|  | </para> | 
|  | <para> | 
|  | Examples: fail to access a resource required by the | 
|  | app, etc. | 
|  | </para> | 
|  | </listitem> | 
|  | </varlistentry> | 
|  | <varlistentry> | 
|  | <term><literal>TRACE</literal></term> | 
|  | <listitem> | 
|  | <para> | 
|  | These are detailed debugging messages that are mainly | 
|  | useful  to debug a component. These are usually turned | 
|  | off. | 
|  | </para> | 
|  | <para> | 
|  | Examples: everything else that does not fall in one of | 
|  | the above mentioned categories and the user does not | 
|  | need to know about it. | 
|  | </para> | 
|  | </listitem> | 
|  | </varlistentry> | 
|  | </variablelist> | 
|  |  | 
|  | <para> | 
|  | The user has the capability to turn on or off messages of a | 
|  | particular type. You can expect the following patterns of | 
|  | usage (but note that  any combination is possible): | 
|  | </para> | 
|  | <itemizedlist> | 
|  | <listitem> | 
|  | <para> | 
|  | when you debug a component, all types | 
|  | (<literal>TRACE</literal>, <literal>WARN</literal>, | 
|  | <literal>ERR</literal>, <literal>FIXME</literal>) will | 
|  | be enabled. | 
|  | </para> | 
|  | </listitem> | 
|  | <listitem> | 
|  | <para> | 
|  | during the pre-alpha (maybe alpha) stage of Wine, most | 
|  | likely the <literal>TRACE</literal> class will be | 
|  | disabled by default, but all others | 
|  | (<literal>WARN</literal>, <literal>ERR</literal>, | 
|  | <literal>FIXME</literal>) will be enabled by default. | 
|  | </para> | 
|  | </listitem> | 
|  | <listitem> | 
|  | <para> | 
|  | when Wine will become stable, most likely the | 
|  | <literal>TRACE</literal> and <literal>WARN</literal> | 
|  | classes will be disabled by default, but all | 
|  | <literal>ERR</literal>s and <literal>FIXME</literal>s | 
|  | will be enabled. | 
|  | </para> | 
|  | </listitem> | 
|  | <listitem> | 
|  | <para> | 
|  | in some installations that want the smallest footprint | 
|  | and where the debug information is of no interest,  all | 
|  | classes may be disabled by default. | 
|  | </para> | 
|  | </listitem> | 
|  | </itemizedlist> | 
|  | <para> | 
|  | Of course, the user will have the runtime ability to | 
|  | override these defaults. However, this ability may be turned | 
|  | off and certain classes of messages may be completely | 
|  | disabled at compile time to reduce the  size of Wine. | 
|  | </para> | 
|  | </sect2> | 
|  |  | 
|  | <sect2> | 
|  | <title>Debugging channels</title> | 
|  |  | 
|  | <para> | 
|  | Also, we divide the debugging messages on a component basis. | 
|  | Each component is assigned a debugging channel. The | 
|  | identifier of the channel must be a valid C identifier but | 
|  | note that it may also be a reserved word like | 
|  | <type>int</type> or <type>static</type>. | 
|  | </para> | 
|  | <para> | 
|  | Examples of debugging channels: | 
|  | <simplelist type="inline"> | 
|  | <member><literal>reg</literal></member> | 
|  | <member><literal>updown</literal></member> | 
|  | <member><literal>string</literal></member> | 
|  | </simplelist> | 
|  | </para> | 
|  | <para> | 
|  | We will refer to a generic channel as <literal>xxx</literal>. | 
|  | </para> | 
|  | <note> | 
|  | <para> | 
|  | for those who know the old interface, the channel/type is | 
|  | what followed the _ in the | 
|  | <function>dprintf_xxx</function> statements. For example, | 
|  | to output a message on the debugging channel | 
|  | <literal>reg</literal> in the old interface you would had | 
|  | to write: | 
|  | </para> | 
|  | <programlisting> | 
|  | dprintf_reg(stddeb, "Could not access key!\n"); | 
|  | </programlisting> | 
|  | <para> | 
|  | In the new interface, we drop the | 
|  | <literal>stddeb</literal> as it is implicit. However, we | 
|  | add an orthogonal piece of information to the message: its | 
|  | class. This is very important as it will allow us to | 
|  | selectively turn on or off certain messages based on the | 
|  | type of information they report. For this reason it is | 
|  | essential to choose the right class for the message. | 
|  | Anyhow, suppose we figured that this message should belong | 
|  | in the <literal>WARN</literal> class, so in the new | 
|  | interface, you write: | 
|  | </para> | 
|  | <programlisting> | 
|  | WARN(reg, "Could not access key!\n"); | 
|  | </programlisting> | 
|  | </note> | 
|  | </sect2> | 
|  |  | 
|  | <sect2> | 
|  | <title>How to use it</title> | 
|  |  | 
|  | <para> | 
|  | So, to output a message (class <literal>YYY</literal>) on | 
|  | channel <literal>xxx</literal>, do: | 
|  | </para> | 
|  | <programlisting> | 
|  | #include "debug.h" | 
|  |  | 
|  | .... | 
|  |  | 
|  | YYY(xxx, "<message>", ...); | 
|  | </programlisting> | 
|  | <para> | 
|  | Some examples from the code: | 
|  | </para> | 
|  | <programlisting> | 
|  | #include "debug.h" | 
|  |  | 
|  | ... | 
|  |  | 
|  | TRACE(crtdll, "CRTDLL_setbuf(file %p buf %p)", file, buf); | 
|  |  | 
|  | WARN(aspi, "Error opening device errno=%d", save_error); | 
|  | </programlisting> | 
|  | <para> | 
|  | If you need to declare a new debugging channel, use it in | 
|  | your code and then do: | 
|  | </para> | 
|  | <screen> | 
|  | %tools/make_debug | 
|  | </screen> | 
|  | <para> | 
|  | in the root directory of Wine. Note that this will result in | 
|  | almost complete recompilation of Wine. | 
|  | </para> | 
|  |  | 
|  | <note> | 
|  | <orderedlist> | 
|  | <listitem> | 
|  | <para> | 
|  | Please pay attention to which class you assign the | 
|  | message. There are only 4 classes, so it is not hard. | 
|  | The reason it is important to get it right is that too | 
|  | much information is no information. For example, if | 
|  | you put things into the <literal>WARN</literal> class | 
|  | that should really be in the <literal>TRACE</literal> | 
|  | class, the  output will be too big and this will force | 
|  | the user to  turn warnings off. But this way he will | 
|  | fail to see the important ones. Also, if you put | 
|  | warnings into the <literal>TRACE</literal> class lets | 
|  | say, he will most likely miss those because usually | 
|  | the <literal>TRACE</literal> class is turned off. A | 
|  | similar argument can be made if you mix any other two | 
|  | classes. | 
|  | </para> | 
|  | </listitem> | 
|  | <listitem> | 
|  | <para> | 
|  | All lines should end with a newline. If you can NOT | 
|  | output everything that you want in the line with only | 
|  | one statement, then you need to build the string in | 
|  | memory. Please read the section below "In-memory | 
|  | messages" on the preferred way to do it. PLEASE USE | 
|  | THAT INTERFACE TO BUILD MESSAGES IN MEMORY. The reason | 
|  | is that we are not sure that we like it and having | 
|  | everything in one format will facilitate the | 
|  | (automatic) translation to a better interface. | 
|  | </para> | 
|  | </listitem> | 
|  | </orderedlist> | 
|  | </note> | 
|  | </sect2> | 
|  |  | 
|  | <sect2> | 
|  | <title>Are we debugging?</title> | 
|  |  | 
|  | <para> | 
|  | To test whether the debugging output of class | 
|  | <literal>yyy</literal> on channel <literal>xxx</literal> is | 
|  | enabled, use: | 
|  | </para> | 
|  | <screen> | 
|  | TRACE_ON  to test if TRACE is enabled | 
|  | WARN_ON   to test if WARN is enabled | 
|  | FIXME_ON  to test if FIXME is enabled | 
|  | ERR_ON    to test if ERR is enabled | 
|  | </screen> | 
|  | <para> | 
|  | Examples: | 
|  | </para> | 
|  | <programlisting> | 
|  | if(TRACE_ON(atom)){ | 
|  | ...blah... | 
|  | } | 
|  | </programlisting> | 
|  |  | 
|  | <note> | 
|  | <para> | 
|  | You should normally need to test only if | 
|  | <literal>TRACE_ON</literal>. At present, none of the other | 
|  | 3 tests (except for <literal>ERR_ON</literal> which is | 
|  | used only once!) are used in Wine. | 
|  | </para> | 
|  | </note> | 
|  | </sect2> | 
|  |  | 
|  | <sect2> | 
|  | <title>In-memory messages</title> | 
|  |  | 
|  | <para> | 
|  | If you NEED to build the message from multiple calls, you | 
|  | need to  build it in memory. To do that, you should use the | 
|  | following interface: | 
|  | </para> | 
|  |  | 
|  | <orderedlist> | 
|  | <listitem> | 
|  | <para> | 
|  | declare a string (where you are allowed to declare C | 
|  | variables) as follows: | 
|  | <programlisting> | 
|  | dbg_decl_str(name, len); | 
|  | </programlisting> | 
|  | where <parameter>name</parameter> is the name of the | 
|  | string (you should use the channel name on which you | 
|  | are going to output it) | 
|  | </para> | 
|  | </listitem> | 
|  | <listitem> | 
|  | <para> | 
|  | print in it with: | 
|  | <programlisting> | 
|  | dsprintf(name, "<message>", ...); | 
|  | </programlisting> | 
|  | which is just like a <function>sprintf</function> | 
|  | function but instead of a C string as first parameter it | 
|  | takes the name you used to declare it. | 
|  | </para> | 
|  | </listitem> | 
|  | <listitem> | 
|  | <para> | 
|  | obtain a pointer to the string with: <function>dbg_str(name)</function> | 
|  | </para> | 
|  | </listitem> | 
|  | <listitem> | 
|  | <para> | 
|  | reset the string (if you want to reuse it with): | 
|  | <programlisting> | 
|  | dbg_reset_str(name); | 
|  | </programlisting> | 
|  | </para> | 
|  | </listitem> | 
|  | </orderedlist> | 
|  |  | 
|  | <para> | 
|  | Example (modified from the code): | 
|  | </para> | 
|  | <programlisting> | 
|  | void some_func(tabs) | 
|  | { | 
|  | INT32 i; | 
|  | LPINT16 p = (LPINT16)tabs; | 
|  | dbg_decl_str(listbox, 256);                   /* declare the string */ | 
|  |  | 
|  | for (i = 0; i < descr->nb_tabs; i++) { | 
|  | descr->tabs[i] = *p++<<1; | 
|  | if(TRACING(listbox))                         /* write in it only if | 
|  | dsprintf(listbox, "%hd ", descr->tabs[i]); /* we are gonna output it */ | 
|  | } | 
|  | TRACE(listbox, "Listbox %04x: settabstops %s", | 
|  | wnd->hwndSelf, dbg_str(listbox));        /* output the whole thing */ | 
|  | } | 
|  | </programlisting> | 
|  | <para> | 
|  | If you need to use it two times in the same scope do like | 
|  | this: | 
|  | </para> | 
|  | <programlisting> | 
|  | void some_func(tabs) | 
|  | { | 
|  | INT32 i; | 
|  | LPINT16 p = (LPINT16)tabs; | 
|  | dbg_decl_str(listbox, 256);                   /* declare the string      */ | 
|  |  | 
|  | for (i = 0; i < descr->nb_tabs; i++) { | 
|  | descr->tabs[i] = *p++<<1; | 
|  | if(TRACING(listbox))                         /* write in it only if | 
|  | dsprintf(listbox, "%hd ", descr->tabs[i]); /* we are gonna output it */ | 
|  | } | 
|  | TRACE(listbox, "Listbox %04x: settabstops %s\n", | 
|  | wnd->hwndSelf, dbg_str(listbox));        /* output the whole thing */ | 
|  |  | 
|  | dbg_reset_str(listbox);                        /* !!!reset the string!!! */ | 
|  | for (i = 0; i < descr->extrainfo_nr; i++) { | 
|  | descr->extrainfo = *p+1; | 
|  | if(TRACING(listbox))                         /* write in it only if | 
|  | dsprintf(listbox,"%3d ",descr->extrainfo); /* we are gonna output it */ | 
|  | } | 
|  |  | 
|  | TRACE(listbox, "Listbox %04x: extrainfo %s\n", | 
|  | wnd->hwndSelf, dbg_str(listbox));        /* output the whole thing */ | 
|  |  | 
|  | } | 
|  | </programlisting> | 
|  |  | 
|  | <important> | 
|  | <para> | 
|  | As I already stated, I do not think this will be the | 
|  | ultimate interface for building in-memory debugging | 
|  | messages. In fact, I do have better ideas which I hope to | 
|  | have time to implement for the next release. For this | 
|  | reason, please try not to use it. However, if you need to | 
|  | output a line in more than one | 
|  | <function>dprintf_xxx</function> calls, then USE THIS | 
|  | INTERFACE. DO NOT use other methods. This way, I will | 
|  | easily translate everything to the new interface (when it | 
|  | will become available). So, if you need to use it, then | 
|  | follow the following guidelines: | 
|  | </para> | 
|  | <itemizedlist> | 
|  | <listitem> | 
|  | <para>wrap calls to <function>dsprintf</function> with a | 
|  | </para> | 
|  | <programlisting> | 
|  | if(YYY(xxx)) | 
|  | dsprintf(xxx,...); | 
|  | </programlisting> | 
|  | <para> | 
|  | Of course, if the call to | 
|  | <function>dsprintf</function> is made from within a | 
|  | function  which you know is called only if | 
|  | <function>YYY(xxx)</function> is true, for example if | 
|  | you call it only like this: | 
|  | </para> | 
|  | <programlisting> | 
|  | if(YYY(xxx)) | 
|  | print_some_debug_info(); | 
|  | </programlisting> | 
|  | <para> | 
|  | then you need not (and should not) wrap calls to | 
|  | <function>dsprintf</function> with the before | 
|  | mentioned <function>if</function>. | 
|  | </para> | 
|  | </listitem> | 
|  | <listitem> | 
|  | <para> | 
|  | name the string EXACTLY like the debugging channel on | 
|  | which is going to be output. Please see the above | 
|  | example. | 
|  | </para> | 
|  | </listitem> | 
|  | </itemizedlist> | 
|  | </important> | 
|  | </sect2> | 
|  |  | 
|  | <sect2> | 
|  | <title>Resource identifiers</title> | 
|  |  | 
|  | <para> | 
|  | Resource identifiers can be either strings or numbers. To | 
|  | make life a bit easier for outputting these beasts (and to | 
|  | help you avoid the need to build the message in memory), I | 
|  | introduced a new function called <function>debugres</function>. | 
|  | </para> | 
|  | <para> | 
|  | The function is defined in <filename>debugstr.h</filename> | 
|  | and has the following prototype: | 
|  | </para> | 
|  | <programlisting> | 
|  | LPSTR debugres(const void *id); | 
|  | </programlisting> | 
|  | <para> | 
|  | It takes a pointer to the resource id and returns a nicely | 
|  | formatted string of the identifier. If the high word of the | 
|  | pointer is <literal>0</literal>, then it assumes that the | 
|  | identifier is a number and thus returns a string of the | 
|  | form: | 
|  | </para> | 
|  | <programlisting> | 
|  | #xxxx | 
|  | </programlisting> | 
|  | <para> | 
|  | where <literal>xxxx</literal> are 4 hex-digits representing | 
|  | the low word of <parameter>id</parameter>. | 
|  | </para> | 
|  | <para> | 
|  | If the high word of the pointer is not <literal>0</literal>, | 
|  | then it assumes that the identifier is a string and thus | 
|  | returns a string of the form: | 
|  | </para> | 
|  | <programlisting> | 
|  | '<identifier>' | 
|  | </programlisting> | 
|  | <para> | 
|  | Thus, to use it, do something on the following lines: | 
|  | </para> | 
|  | <programlisting> | 
|  | #include "debug.h" | 
|  |  | 
|  | ... | 
|  |  | 
|  | YYY(xxx, "resource is %s", debugres(myresource)); | 
|  | </programlisting> | 
|  | </sect2> | 
|  |  | 
|  | <sect2> | 
|  | <title>The <parameter>--debugmsg</parameter> command line option</title> | 
|  |  | 
|  | <para> | 
|  | So, the <parameter>--debugmsg</parameter> command line | 
|  | option has been changed as follows: | 
|  | </para> | 
|  | <itemizedlist> | 
|  | <listitem> | 
|  | <para> | 
|  | the new syntax is: <parameter>--debugmsg | 
|  | [yyy]#xxx[,[yyy1]#xxx1]*</parameter> where | 
|  | <literal>#</literal> is either <literal>+</literal> or | 
|  | <literal>-</literal> | 
|  | </para> | 
|  | </listitem> | 
|  | <listitem> | 
|  | <para> | 
|  | when the optional class argument (<literal>yyy</literal>) | 
|  | is not present, then the statement will | 
|  | enable(<literal>+</literal>)/disable(<literal>-</literal>) | 
|  | all messages for the given channel (<literal>xxx</literal>) | 
|  | on all classes. For example: | 
|  | </para> | 
|  | <programlisting> | 
|  | --debugmsg +reg,-file | 
|  | </programlisting> | 
|  | <para> | 
|  | enables all messages on the <literal>reg</literal> | 
|  | channel and disables all messages on the | 
|  | <literal>file</literal> channel. This is same as the old | 
|  | semantics. | 
|  | </para> | 
|  | </listitem> | 
|  | <listitem> | 
|  | <para> | 
|  | when the optional class argument (<literal>yyy</literal>) | 
|  | is present,  then the statement will enable | 
|  | (<literal>+</literal>)/disable(<literal>-</literal>) | 
|  | messages for the given channel (<literal>xxx</literal>) | 
|  | only on the given class. For example: | 
|  | </para> | 
|  | <programlisting> | 
|  | --debugmsg trace+reg,warn-file | 
|  | </programlisting> | 
|  | <para> | 
|  | enables trace messages on the <literal>reg</literal> | 
|  | channel and disables warning messages on the | 
|  | <literal>file</literal> channel. | 
|  | </para> | 
|  | </listitem> | 
|  | <listitem> | 
|  | <para> | 
|  | also, the pseudo-channel all is also supported and it | 
|  | has the  intuitive semantics: | 
|  | </para> | 
|  | <screen> | 
|  | --debugmsg +all      -- enables all debug messages | 
|  | --debugmsg -all      -- disables all debug messages | 
|  | --debugmsg yyy+all   -- enables debug messages for class yyy on all | 
|  | channels. | 
|  | --debugmsg yyy-all   -- disables debug messages for class yyy on all | 
|  | channels. | 
|  | </screen> | 
|  | <para> | 
|  | So, for example: | 
|  | </para> | 
|  | <screen> | 
|  | --debugmsg warn-all  -- disables all warning messages. | 
|  | </screen> | 
|  | </listitem> | 
|  | </itemizedlist> | 
|  |  | 
|  | <para> | 
|  | Also, note that at the moment: | 
|  | </para> | 
|  | <itemizedlist> | 
|  | <listitem> | 
|  | <para>the <literal>FIXME</literal> and <literal>ERR</literal> | 
|  | classes are enabled by default</para> | 
|  | </listitem> | 
|  | <listitem> | 
|  | <para>the <literal>TRACE</literal> and | 
|  | <literal>WARN</literal> classes are disabled by | 
|  | default</para> | 
|  | </listitem> | 
|  | </itemizedlist> | 
|  | </sect2> | 
|  |  | 
|  | <sect2> | 
|  | <title>Compiling Out Debugging Messages</title> | 
|  |  | 
|  | <para> | 
|  | To compile out the debugging messages, provide | 
|  | <command>configure</command> with the following options: | 
|  | </para> | 
|  | <screen> | 
|  | --disable-debug      -- turns off TRACE, WARN, and FIXME (and DUMP). | 
|  | --disable-trace      -- turns off TRACE only. | 
|  | </screen> | 
|  | <para> | 
|  | This will result in an executable that, when stripped, is | 
|  | about 15%-20% smaller.  Note, however, that you will not be | 
|  | able to effectively debug Wine without these messages. | 
|  | </para> | 
|  | <para> | 
|  | This feature has not been extensively tested--it may subtly | 
|  | break some things. | 
|  | </para> | 
|  | </sect2> | 
|  |  | 
|  | <sect2> | 
|  | <title>A Few Notes on Style</title> | 
|  |  | 
|  | <para> | 
|  | This new scheme makes certain things more consistent but | 
|  | there is still room for improvement by using a common style | 
|  | of debug messages. Before I continue, let me note that the | 
|  | output format is the following: | 
|  | </para> | 
|  | <screen> | 
|  | yyy:xxx:fff <message> | 
|  |  | 
|  | where: | 
|  | yyy = the class (fixme, err, warn, trace) | 
|  | xxx = the channel (atom, win, font, etc) | 
|  | fff = the function name | 
|  | </screen> | 
|  | <para> | 
|  | these fields are output automatically. All you have to | 
|  | provide is the <message> part. | 
|  | </para> | 
|  | <para> | 
|  | So here are some ideas: | 
|  | </para> | 
|  |  | 
|  | <itemizedlist> | 
|  | <listitem> | 
|  | <para>do NOT include the name of the function: it is included automatically</para> | 
|  | </listitem> | 
|  | <listitem> | 
|  | <para> | 
|  | if you want to output the parameters of the function, do | 
|  | it as the first thing and include them in parentheses, | 
|  | like this: | 
|  | <programlisting> | 
|  | YYY(xxx, "(%d,%p,etc)...\n", par1, par2, ...); | 
|  | </programlisting> | 
|  | </para> | 
|  | </listitem> | 
|  | <listitem> | 
|  | <para> | 
|  | for stubs, you should output a <literal>FIXME</literal> | 
|  | message. I suggest this style: | 
|  | <programlisting> | 
|  | FIXME(xxx, "(%x,%d...): stub\n", par1, par2, ...); | 
|  | </programlisting> | 
|  | That is, you output the parameters, then a : and then a string | 
|  | containing the word "stub". I've seen "empty stub", and others, but I | 
|  | think that just "stub" suffices. | 
|  | </para> | 
|  | </listitem> | 
|  | <listitem> | 
|  | <para> | 
|  | output 1 and ONLY 1 line per message. That is, the format | 
|  | string should contain only 1 <literal>\n</literal> and it | 
|  | should always appear at the end of the string. (there are | 
|  | many reasons  for this requirement, one of them is that | 
|  | each debug macro adds things to the beginning of the line) | 
|  | </para> | 
|  | </listitem> | 
|  | <listitem> | 
|  | <para> | 
|  | if you want to name a value, use <literal>=</literal> and | 
|  | NOT <literal>:</literal>. That is, instead of saying: | 
|  | <programlisting> | 
|  | FIXME(xxx, "(fd: %d, file: %s): stub\n", fd, name); | 
|  | </programlisting> | 
|  | say: | 
|  | <programlisting> | 
|  | FIXME(xxx, "(fd=%d, file=%s): stub\n", fd, name); | 
|  | </programlisting> | 
|  | use <literal>:</literal> to separate categories. | 
|  | </para> | 
|  | </listitem> | 
|  | <listitem> | 
|  | <para> | 
|  | try to avoid the style: | 
|  | <programlisting> | 
|  | FIXME(xxx, "(fd=%d, file=%s): stub\n", fd, name); | 
|  | </programlisting> | 
|  | but use: | 
|  | <programlisting> | 
|  | FIXME(xxx, "(fd=%d, file=%s): stub\n", fd, name); | 
|  | </programlisting> | 
|  | The reason is that if you want to <command>grep</command> | 
|  | for things, you would search for <literal>FIXME</literal> | 
|  | but in the first case there is no additional information | 
|  | available, where in the second one, there is (e.g. the word | 
|  | stub) | 
|  | </para> | 
|  | </listitem> | 
|  | <listitem> | 
|  | <para> | 
|  | if you output a string s that might contain control | 
|  | characters, or if <parameter>s</parameter> may be | 
|  | <literal>NULL</literal>, use | 
|  | <function>debugstr_a</function> (for ASCII strings, or | 
|  | <function>debugstr_w</function> for Unicode strings) to | 
|  | convert <parameter>s</parameter> to a C string, like  this: | 
|  | <programlisting> | 
|  | HANDLE32 WINAPI YourFunc(LPCSTR s) | 
|  | { | 
|  | FIXME(xxx, "(%s): stub\n", debugstr_a(s)); | 
|  | } | 
|  | </programlisting> | 
|  | </para> | 
|  | </listitem> | 
|  | <listitem> | 
|  | <para> | 
|  | if you want to output a resource identifier, use debugres to | 
|  | convert it to a string first, like this: | 
|  | <programlisting> | 
|  | HANDLE32 WINAPI YourFunc(LPCSTR res) | 
|  | { | 
|  | FIXME(xxx, "(res=%s): stub\n", debugres(s)); | 
|  | } | 
|  | </programlisting> | 
|  | if the resource identifier is a <type>SEGPTR</type>, use | 
|  | <function>PTR_SEG_TO_LIN</function> to get a | 
|  | liner pointer first: | 
|  | <programlisting> | 
|  | HRSRC16 WINAPI FindResource16( HMODULE16 hModule, SEGPTR name, SEGPTR type ) | 
|  | { | 
|  | [...] | 
|  | TRACE(resource, "module=%04x name=%s type=%s\n", | 
|  | hModule, debugres(PTR_SEG_TO_LIN(name)), | 
|  | debugres(PTR_SEG_TO_LIN(type)) ); | 
|  | [...] | 
|  | } | 
|  | </programlisting> | 
|  | </para> | 
|  | </listitem> | 
|  | <listitem> | 
|  | <para> | 
|  | for messages intended for the user (specifically those that | 
|  | report errors in <filename>wine.conf</filename>), use the | 
|  | <literal>MSG</literal> macro. Use it like a | 
|  | <function>printf</function>: | 
|  | <programlisting> | 
|  | MSG( "Definition of drive %d is incorrect!\n", drive ); | 
|  | </programlisting> | 
|  | However, note that there are <emphasis>very</emphasis> few | 
|  | valid uses of this macro. Most messages are debugging | 
|  | messages, so chances are you will not need to use this | 
|  | macro. Grep the source to get an idea where it is | 
|  | appropriate to use it. | 
|  | </para> | 
|  | </listitem> | 
|  | <listitem> | 
|  | <para> | 
|  | For structure dumps, use the <function>DUMP</function> | 
|  | macro. Use it like a <function>printf</function>, just like | 
|  | the <literal>MSG</literal> macro. Similarly, there are only | 
|  | a few valid uses of this macro. Grep the source to see when | 
|  | to use it. | 
|  | </para> | 
|  | </listitem> | 
|  | </itemizedlist> | 
|  | </sect2> | 
|  | </sect1> | 
|  |  | 
|  | <sect1 id="wine-debugger"> | 
|  | <title>Using the Wine Debugger</title> | 
|  |  | 
|  | <para> | 
|  | written by Marcus Meissner <email>msmeissn@cip.informatik.uni-erlangen.de</email>, | 
|  | additions welcome. | 
|  | </para> | 
|  | <para> | 
|  | (Extracted from <filename>wine/documentation/debugging</filename>) | 
|  | </para> | 
|  |  | 
|  | <para> | 
|  | This file describes where to start debugging Wine. If at any | 
|  | point you get stuck and want to ask for help, please read the | 
|  | file <filename>documentation/bugreports</filename> for | 
|  | information on how to write useful bug reports. | 
|  | </para> | 
|  |  | 
|  | <sect2> | 
|  | <title>Crashes</title> | 
|  |  | 
|  | <para> | 
|  | These usually show up like this: | 
|  | </para> | 
|  | <screen> | 
|  | |Unexpected Windows program segfault - opcode = 8b | 
|  | |Segmentation fault in Windows program 1b7:c41. | 
|  | |Loading symbols from ELF file /root/wine/wine... | 
|  | |....more Loading symbols from ... | 
|  | |In 16 bit mode. | 
|  | |Register dump: | 
|  | | CS:01b7 SS:016f DS:0287 ES:0000 | 
|  | | IP:0c41 SP:878a BP:8796 FLAGS:0246 | 
|  | | AX:811e BX:0000 CX:0000 DX:0000 SI:0001 DI:ffff | 
|  | |Stack dump: | 
|  | |0x016f:0x878a:  0001 016f ffed 0000 0000 0287 890b 1e5b | 
|  | |0x016f:0x879a:  01b7 0001 000d 1050 08b7 016f 0001 000d | 
|  | |0x016f:0x87aa:  000a 0003 0004 0000 0007 0007 0190 0000 | 
|  | |0x016f:0x87ba: | 
|  | | | 
|  | |0050: sel=0287 base=40211d30 limit=0b93f (bytes) 16-bit rw- | 
|  | |Backtrace: | 
|  | |0 0x01b7:0x0c41 (PXSRV_FONGETFACENAME+0x7c) | 
|  | |1 0x01b7:0x1e5b (PXSRV_FONPUTCATFONT+0x2cd) | 
|  | |2 0x01a7:0x05aa | 
|  | |3 0x01b7:0x0768 (PXSRV_FONINITFONTS+0x81) | 
|  | |4 0x014f:0x03ed (PDOXWIN_@SQLCURCB$Q6CBTYPEULN8CBSCTYPE+0x1b1) | 
|  | |5 0x013f:0x00ac | 
|  | | | 
|  | |0x01b7:0x0c41 (PXSRV_FONGETFACENAME+0x7c):  movw        %es:0x38(%bx),%dx | 
|  | </screen> | 
|  | <para> | 
|  | Steps to debug a crash. You may stop at any step, but please | 
|  | report the bug and provide as much of the information | 
|  | gathered to the newsgroup or the relevant developer as | 
|  | feasible. | 
|  | </para> | 
|  |  | 
|  | <orderedlist> | 
|  | <listitem> | 
|  | <para> | 
|  | Get the reason for the crash. This is usually an access to | 
|  | an invalid selector, an access to an out of range address | 
|  | in a valid selector, popping a segmentregister from the | 
|  | stack or the like. When reporting a crash, report this | 
|  | <emphasis>whole</emphasis> crashdump even if it doesn't | 
|  | make sense to you. | 
|  | </para> | 
|  | <para> | 
|  | (In this case it is access to an invalid selector, for | 
|  | <systemitem>%es</systemitem> is <literal>0000</literal>, as | 
|  | seen in the register dump). | 
|  | </para> | 
|  | </listitem> | 
|  | <listitem> | 
|  | <para> | 
|  | Determine the cause of the crash. Since this is usually | 
|  | a primary/secondary reaction to a failed or misbehaving | 
|  | Wine function, rerun Wine with <parameter>-debugmsg | 
|  | +relay</parameter> added to the commandline. This will | 
|  | generate quite a lot of output, but usually the reason is | 
|  | located in the last call(s).  Those lines usually look like | 
|  | this: | 
|  | </para> | 
|  | <screen> | 
|  | |Call KERNEL.90: LSTRLEN(0227:0692 "text") ret=01e7:2ce7 ds=0227 | 
|  | ^^^^^^^^^  ^       ^^^^^^^^^ ^^^^^^      ^^^^^^^^^    ^^^^ | 
|  | |          |       |         |           |            |Datasegment | 
|  | |          |       |         |           |Return address | 
|  | |          |       |         |textual parameter | 
|  | |          |       | | 
|  | |          |       |Argument(s). This one is a win16 segmented pointer. | 
|  | |          |Function called. | 
|  | |The module, the function is called in. In this case it is KERNEL. | 
|  |  | 
|  | |Ret  KERNEL.90: LSTRLEN() retval=0x0004 ret=01e7:2ce7 ds=0227 | 
|  | ^^^^^^ | 
|  | |Returnvalue is 16 bit and has the value 4. | 
|  | </screen> | 
|  | </listitem> | 
|  | <listitem> | 
|  | <para> | 
|  | If you have found a misbehaving function, try to find out | 
|  | why it misbehaves. Find the function in the source code. | 
|  | Try to make sense of the arguments passed. Usually there is | 
|  | a <function>TRACE(<channel>,"(...)\n");</function> at | 
|  | the beginning of the function. Rerun wine with | 
|  | <parameter>-debugmsg +xyz,+relay</parameter> added to the | 
|  | commandline. | 
|  | </para> | 
|  | </listitem> | 
|  | <listitem> | 
|  | <para> | 
|  | Additional information on how to debug using the internal | 
|  | debugger can be  found in | 
|  | <filename>debugger/README</filename>. | 
|  | </para> | 
|  | </listitem> | 
|  | <listitem> | 
|  | <para> | 
|  | If this information isn't clear enough or if you want to | 
|  | know more about what's happening in the function itself, | 
|  | try running wine with <parameter>-debugmsg | 
|  | +all</parameter>, which dumps ALL included debug | 
|  | information in wine. | 
|  | </para> | 
|  | </listitem> | 
|  | <listitem> | 
|  | <para> | 
|  | If even that isn't enough, add more debug output for | 
|  | yourself into the functions you find relevant.  See | 
|  | <filename>documentation/debug-msgs</filename>. You might | 
|  | also try to run the program in <command>gdb</command> | 
|  | instead of using the WINE-debugger. If you do that, use | 
|  | <parameter>handle SIGSEGV nostop noprint</parameter> to | 
|  | disable the handling of seg faults inside | 
|  | <command>gdb</command> (needed for Win16). If you don't use | 
|  | the <parameter>--desktop</parameter> or | 
|  | <parameter>--managed</parameter> option, start the WINE | 
|  | process with <parameter>--sync</parameter>, or chances are | 
|  | good to get X into an unusable state. | 
|  | </para> | 
|  | </listitem> | 
|  | <listitem> | 
|  | <para> | 
|  | You can also set a breakpoint for that function. Start wine | 
|  | with the <parameter>--debug</parameter> option added to the | 
|  | commandline. After loading the executable wine will enter | 
|  | the internal debugger. Use <parameter>break | 
|  | KERNEL_LSTRLEN</parameter> (replace by function you want | 
|  | to debug, CASE IS RELEVANT) to set a breakpoint.  Then use | 
|  | <command>continue</command> to start normal | 
|  | program-execution. Wine will stop if it reaches the | 
|  | breakpoint. If the program isn't yet at the crashing call | 
|  | of that function, use <command>continue</command> again | 
|  | until you are about to enter that function. You may now | 
|  | proceed with single-stepping the function until you reach | 
|  | the point of crash. Use the other debugger commands to | 
|  | print registers and the like. | 
|  | </para> | 
|  | </listitem> | 
|  | </orderedlist> | 
|  | </sect2> | 
|  |  | 
|  | <sect2> | 
|  | <title>Program hangs, nothing happens</title> | 
|  |  | 
|  | <para> | 
|  | Switch to UNIX shell, get the process-ID using <command>ps -a | | 
|  | grep wine</command>, and do a <command>kill -HUP | 
|  | <pid></command> (without the < and >). Wine will | 
|  | then enter its internal debugger and you can proceed as | 
|  | explained above. Also, you can use | 
|  | <parameter>--debug</parameter> switch and then you can get into | 
|  | internal debugger by pressing | 
|  | <keycombo><keycap>Ctrl</keycap><keycap>C</keycap></keycombo> in | 
|  | the terminal where you run Wine. | 
|  | </para> | 
|  | </sect2> | 
|  |  | 
|  | <sect2> | 
|  | <title>Program reports an error with a Messagebox</title> | 
|  |  | 
|  | <para> | 
|  | Sometimes programs are reporting failure using more or | 
|  | less nondescript messageboxes. We can debug this using the | 
|  | same method as Crashes, but there is one problem... For | 
|  | setting up a message box the program also calls Wine | 
|  | producing huge chunks of debug code. | 
|  | </para> | 
|  | <para> | 
|  | Since the failure happens usually directly before setting up | 
|  | the Messagebox you can start wine with | 
|  | <parameter>--debug</parameter> added to the commandline, set a | 
|  | breakpoint at <function>MessageBoxA</function> (called by win16 | 
|  | and win32 programs) and proceed with | 
|  | <command>continue</command>. With <parameter>--debugmsg | 
|  | +all</parameter> Wine will now stop directly before setting | 
|  | up the Messagebox. Proceed as explained above. | 
|  | </para> | 
|  | <para> | 
|  | You can also run wine using <command>wine -debugmsg +relay | 
|  | program.exe 2>&1 | less -i</command> and in | 
|  | <command>less</command> search for <quote>MessageBox</quote>. | 
|  | </para> | 
|  | </sect2> | 
|  |  | 
|  | <sect2> | 
|  | <title>Disassembling programs:</title> | 
|  |  | 
|  | <para> | 
|  | You may also try to disassemble the offending program to | 
|  | check for undocumented features and/or use of them. | 
|  | </para> | 
|  | <para> | 
|  | The best, freely available, disassembler for Win16 programs is | 
|  | <application>Windows Codeback</application>, archivename | 
|  | <filename>wcbxxx.zip</filename>, which usually can be found in | 
|  | the <filename>Cica-Mirror</filename> subdirectory on the WINE | 
|  | ftpsites. (See <filename>ANNOUNCE</filename>). | 
|  | </para> | 
|  | <para> | 
|  | Disassembling win32 programs is possible using | 
|  | <application>Windows Disassembler 32</application>, archivename | 
|  | something like <filename>w32dsm87.zip</filename> (or similar) | 
|  | on <systemitem class="systemname">ftp.winsite.com</systemitem> | 
|  | and mirrors.  The shareware version does not allow saving of | 
|  | disassembly listings. You can also use the newer (and in the | 
|  | full version better) <application>Interactive | 
|  | Disassembler</application> (IDA) from the ftp sites mentioned | 
|  | at the end of the document. Understanding disassembled code is | 
|  | mostly a question of exercise. | 
|  | </para> | 
|  | <para> | 
|  | Most code out there uses standard C function entries (for it | 
|  | is usually  written in C). Win16 function entries usually | 
|  | look like that: | 
|  | </para> | 
|  | <programlisting> | 
|  | push bp | 
|  | mov bp, sp | 
|  | ... function code .. | 
|  | retf XXXX 	<--------- XXXX is number of bytes of arguments | 
|  | </programlisting> | 
|  | <para> | 
|  | This is a <function>FAR</function> function with no local | 
|  | storage. The arguments usually start at | 
|  | <literal>[bp+6]</literal> with increasing offsets. Note, that | 
|  | <literal>[bp+6]</literal> belongs to the | 
|  | <emphasis>rightmost</emphasis> argument, for exported win16 | 
|  | functions use the PASCAL calling convention. So, if we use | 
|  | <function>strcmp(a,b)</function> with <parameter>a</parameter> | 
|  | and <parameter>b</parameter> both 32 bit variables | 
|  | <parameter>b</parameter> would be at <literal>[bp+6]</literal> | 
|  | and <parameter>a</parameter> at <literal>[bp+10]</literal>. | 
|  | </para> | 
|  | <para> | 
|  | Most functions make also use of local storage in the stackframe: | 
|  | </para> | 
|  | <programlisting> | 
|  | enter 0086, 00 | 
|  | ... function code ... | 
|  | leave | 
|  | retf XXXX | 
|  | </programlisting> | 
|  | <para> | 
|  | This does mostly the same as above, but also adds | 
|  | <literal>0x86</literal> bytes of stackstorage, which is | 
|  | accessed using <literal>[bp-xx]</literal>. Before calling a | 
|  | function, arguments are pushed on the stack using something | 
|  | like this: | 
|  | </para> | 
|  | <programlisting> | 
|  | push word ptr [bp-02]	<- will be at [bp+8] | 
|  | push di			<- will be at [bp+6] | 
|  | call KERNEL.LSTRLEN | 
|  | </programlisting> | 
|  | <para> | 
|  | Here first the selector and then the offset to the passed | 
|  | string are pushed. | 
|  | </para> | 
|  | </sect2> | 
|  |  | 
|  | <sect2> | 
|  | <title>Sample debugging session:</title> | 
|  |  | 
|  | <para> | 
|  | Let's debug the infamous Word <filename>SHARE.EXE</filename> | 
|  | messagebox: | 
|  | </para> | 
|  | <screen> | 
|  | |marcus@jet $ wine winword.exe | 
|  | |            +---------------------------------------------+ | 
|  | |            | !  You must leave Windows and load SHARE.EXE| | 
|  | |            |    before starting Word.                    | | 
|  | |            +---------------------------------------------+ | 
|  | </screen> | 
|  | <screen> | 
|  | |marcus@jet $ wine winword.exe -debugmsg +relay -debug | 
|  | |CallTo32(wndproc=0x40065bc0,hwnd=000001ac,msg=00000081,wp=00000000,lp=00000000) | 
|  | |Win16 task 'winword': Breakpoint 1 at 0x01d7:0x001a | 
|  | |CallTo16(func=0127:0070,ds=0927) | 
|  | |Call WPROCS.24: TASK_RESCHEDULE() ret=00b7:1456 ds=0927 | 
|  | |Ret  WPROCS.24: TASK_RESCHEDULE() retval=0x8672 ret=00b7:1456 ds=0927 | 
|  | |CallTo16(func=01d7:001a,ds=0927) | 
|  | |     AX=0000 BX=3cb4 CX=1f40 DX=0000 SI=0000 DI=0927 BP=0000 ES=11f7 | 
|  | |Loading symbols: /home/marcus/wine/wine... | 
|  | |Stopped on breakpoint 1 at 0x01d7:0x001a | 
|  | |In 16 bit mode. | 
|  | |Wine-dbg>break MessageBoxA                          <---- Set Breakpoint | 
|  | |Breakpoint 2 at 0x40189100 (MessageBoxA [msgbox.c:190]) | 
|  | |Wine-dbg>c                                            <---- Continue | 
|  | |Call KERNEL.91: INITTASK() ret=0157:0022 ds=08a7 | 
|  | |     AX=0000 BX=3cb4 CX=1f40 DX=0000 SI=0000 DI=08a7 ES=11d7 EFL=00000286 | 
|  | |CallTo16(func=090f:085c,ds=0dcf,0x0000,0x0000,0x0000,0x0000,0x0800,0x0000,0x0000,0x0dcf) | 
|  | |...                                                   <----- Much debugoutput | 
|  | |Call KERNEL.136: GETDRIVETYPE(0x0000) ret=060f:097b ds=0927 | 
|  | ^^^^^^ Drive 0 (A:) | 
|  | |Ret  KERNEL.136: GETDRIVETYPE() retval=0x0002 ret=060f:097b ds=0927 | 
|  | ^^^^^^  DRIVE_REMOVEABLE | 
|  | (It is a floppy diskdrive.) | 
|  |  | 
|  | |Call KERNEL.136: GETDRIVETYPE(0x0001) ret=060f:097b ds=0927 | 
|  | ^^^^^^ Drive 1 (B:) | 
|  | |Ret  KERNEL.136: GETDRIVETYPE() retval=0x0000 ret=060f:097b ds=0927 | 
|  | ^^^^^^  DRIVE_CANNOTDETERMINE | 
|  | (I don't have drive B: assigned) | 
|  |  | 
|  | |Call KERNEL.136: GETDRIVETYPE(0x0002) ret=060f:097b ds=0927 | 
|  | ^^^^^^^ Drive 2 (C:) | 
|  | |Ret  KERNEL.136: GETDRIVETYPE() retval=0x0003 ret=060f:097b ds=0927 | 
|  | ^^^^^^ DRIVE_FIXED | 
|  | (specified as a harddisk) | 
|  |  | 
|  | |Call KERNEL.97: GETTEMPFILENAME(0x00c3,0x09278364"doc",0x0000,0927:8248) ret=060f:09b1 ds=0927 | 
|  | ^^^^^^           ^^^^^        ^^^^^^^^^ | 
|  | |                |            |buffer for fname | 
|  | |                |temporary name ~docXXXX.tmp | 
|  | |Force use of Drive C:. | 
|  |  | 
|  | |Warning: GetTempFileName returns 'C:~doc9281.tmp', which doesn't seem to be writeable. | 
|  | |Please check your configuration file if this generates a failure. | 
|  | </screen> | 
|  | <para> | 
|  | Whoops, it even detects that something is wrong! | 
|  | </para> | 
|  | <screen> | 
|  | |Ret  KERNEL.97: GETTEMPFILENAME() retval=0x9281 ret=060f:09b1 ds=0927 | 
|  | ^^^^^^ Temporary storage ID | 
|  |  | 
|  | |Call KERNEL.74: OPENFILE(0x09278248"C:~doc9281.tmp",0927:82da,0x1012) ret=060f:09d8 ds=0927 | 
|  | ^^^^^^^^^^^^^^^^ ^^^^^^^^^ ^^^^^^^ | 
|  | |filename        |OFSTRUCT |open mode: | 
|  |  | 
|  | OF_CREATE|OF_SHARE_EXCLUSIVE|OF_READWRITE | 
|  | </screen> | 
|  | <para> | 
|  | This fails, since my <medialabel>C:</medialabel> drive is in | 
|  | this case mounted readonly. | 
|  | </para> | 
|  | <screen> | 
|  | |Ret  KERNEL.74: OPENFILE() retval=0xffff ret=060f:09d8 ds=0927 | 
|  | ^^^^^^ HFILE_ERROR16, yes, it failed. | 
|  |  | 
|  | |Call USER.1: MESSAGEBOX(0x0000,0x09278376"Sie mussen Windows verlassen und SHARE.EXE laden bevor Sie Word starten.",0x00000000,0x1030) ret=060f:084f ds=0927 | 
|  | </screen> | 
|  | <para> | 
|  | And MessageBox'ed. | 
|  | </para> | 
|  | <screen> | 
|  | |Stopped on breakpoint 2 at 0x40189100 (MessageBoxA [msgbox.c:190]) | 
|  | |190     {		<- the sourceline | 
|  | In 32 bit mode. | 
|  | Wine-dbg> | 
|  | </screen> | 
|  | <para> | 
|  | The code seems to find a writeable harddisk and tries to create | 
|  | a file there. To work around this bug, you can define | 
|  | <medialabel>C:</medialabel> as a networkdrive, which is ignored | 
|  | by the code above. | 
|  | </para> | 
|  | </sect2> | 
|  |  | 
|  | <sect2> | 
|  | <title>Debugging Tips</title> | 
|  |  | 
|  | <para> | 
|  | Here are some useful debugging tips, added by Andreas Mohr: | 
|  | </para> | 
|  |  | 
|  | <itemizedlist> | 
|  | <listitem> | 
|  | <para> | 
|  | If you have a program crashing at such an early loader phase that you can't | 
|  | use the Wine debugger normally, but Wine already executes the program's | 
|  | start code, then you may use a special trick. You should do a | 
|  | <programlisting> | 
|  | wine --debugmsg +relay program | 
|  | </programlisting> | 
|  | to get a listing of the functions the program calls in its start function. | 
|  | Now you do a | 
|  | <programlisting> | 
|  | wine --debug winfile.exe | 
|  | </programlisting> | 
|  | </para> | 
|  | <para> | 
|  | This way, you get into <command>Wine-dbg</command>. Now you | 
|  | can set a breakpoint on any function the program calls in | 
|  | the start function and just type <userinput>c</userinput> | 
|  | to bypass the eventual calls of Winfile to this function | 
|  | until you are finally at the place where this function gets | 
|  | called by the crashing start function. Now you can proceed | 
|  | with your debugging as usual. | 
|  | </para> | 
|  | </listitem> | 
|  | <listitem> | 
|  | <para> | 
|  | If you try to run a program and it quits after showing an error messagebox, | 
|  | the problem can usually be identified in the return value of one of the | 
|  | functions executed before <function>MessageBox()</function>. | 
|  | That's why you should re-run the program with e.g. | 
|  | <programlisting> | 
|  | wine --debugmsg +relay <program name> &>relmsg | 
|  | </programlisting> | 
|  | Then do a <command>more relmsg</command> and search for the | 
|  | last occurrence of a call to the string "MESSAGEBOX". This is a line like | 
|  | <programlisting> | 
|  | Call USER.1: MESSAGEBOX(0x0000,0x01ff1246 "Runtime error 219 at 0004:1056.",0x00000000,0x1010) ret=01f7:2160 ds=01ff | 
|  | </programlisting> | 
|  | In my example the lines before the call to | 
|  | <function>MessageBox()</function> look like that: | 
|  | <programlisting> | 
|  | Call KERNEL.96: FREELIBRARY(0x0347) ret=01cf:1033 ds=01ff | 
|  | CallTo16(func=033f:0072,ds=01ff,0x0000) | 
|  | Ret  KERNEL.96: FREELIBRARY() retval=0x0001 ret=01cf:1033 ds=01ff | 
|  | Call KERNEL.96: FREELIBRARY(0x036f) ret=01cf:1043 ds=01ff | 
|  | CallTo16(func=0367:0072,ds=01ff,0x0000) | 
|  | Ret  KERNEL.96: FREELIBRARY() retval=0x0001 ret=01cf:1043 ds=01ff | 
|  | Call KERNEL.96: FREELIBRARY(0x031f) ret=01cf:105c ds=01ff | 
|  | CallTo16(func=0317:0072,ds=01ff,0x0000) | 
|  | Ret  KERNEL.96: FREELIBRARY() retval=0x0001 ret=01cf:105c ds=01ff | 
|  | Call USER.171: WINHELP(0x02ac,0x01ff05b4 "COMET.HLP",0x0002,0x00000000) ret=01cf:1070 ds=01ff | 
|  | CallTo16(func=0117:0080,ds=01ff) | 
|  | Call WPROCS.24: TASK_RESCHEDULE() ret=00a7:0a2d ds=002b | 
|  | Ret  WPROCS.24: TASK_RESCHEDULE() retval=0x0000 ret=00a7:0a2d ds=002b | 
|  | Ret  USER.171: WINHELP() retval=0x0001 ret=01cf:1070 ds=01ff | 
|  | Call KERNEL.96: FREELIBRARY(0x01be) ret=01df:3e29 ds=01ff | 
|  | Ret  KERNEL.96: FREELIBRARY() retval=0x0000 ret=01df:3e29 ds=01ff | 
|  | Call KERNEL.52: FREEPROCINSTANCE(0x02cf00ba) ret=01f7:1460 ds=01ff | 
|  | Ret  KERNEL.52: FREEPROCINSTANCE() retval=0x0001 ret=01f7:1460 ds=01ff | 
|  | Call USER.1: MESSAGEBOX(0x0000,0x01ff1246 "Runtime error 219 at 0004:1056.",0x00000000,0x1010) ret=01f7:2160 ds=01ff | 
|  | </programlisting> | 
|  | </para> | 
|  | <para> | 
|  | I think that the call to <function>MessageBox()</function> | 
|  | in this example is <emphasis>not</emphasis> caused by a | 
|  | wrong result value of some previously executed function | 
|  | (it's happening quite often like that), but instead the | 
|  | messagebox complains about a runtime error at | 
|  | <literal>0x0004:0x1056</literal>. | 
|  | </para> | 
|  | <para> | 
|  | As the segment value of the address is only | 
|  | <literal>4</literal>, I think that that is only an internal | 
|  | program value. But the offset address reveals something | 
|  | quite interesting: Offset <literal>1056</literal> is | 
|  | <emphasis>very</emphasis> close to the return address of | 
|  | <function>FREELIBRARY()</function>: | 
|  | <programlisting> | 
|  | Call KERNEL.96: FREELIBRARY(0x031f) ret=01cf:105c ds=01ff | 
|  | ^^^^ | 
|  | </programlisting> | 
|  | </para> | 
|  | <para> | 
|  | Provided that segment <literal>0x0004</literal> is indeed segment | 
|  | <literal>0x1cf</literal>, we now we can use IDA                 (available at | 
|  | <ulink url="ftp://ftp.uni-koeln.de/pc/msdos/programming/assembler/ida35bx.zip"> | 
|  | ftp://ftp.uni-koeln.de/pc/msdos/programming/assembler/ida35bx.zip</ulink>) to | 
|  | disassemble the part that caused the error. We just have to find the address of | 
|  | the call to <function>FreeLibrary()</function>. Some lines before that the | 
|  | runtime error occurred. But be careful! In some cases you don't have to | 
|  | disassemble the main program, but instead some DLL called by it in order to find | 
|  | the correct place where the runtime error occurred. That can be determined by | 
|  | finding the origin of the segment value (in this case <literal>0x1cf</literal>). | 
|  | </para> | 
|  | </listitem> | 
|  | <listitem> | 
|  | <para> | 
|  | If you have created a relay file of some crashing | 
|  | program and want to set a breakpoint at a certain | 
|  | location which is not yet available as the program loads | 
|  | the breakpoint's segment during execution, you may set a | 
|  | breakpoint to <function>GetVersion16/32</function> as | 
|  | those functions are called very often. | 
|  | </para> | 
|  | <para> | 
|  | Then do a <userinput>c</userinput> until you are able to | 
|  | set this breakpoint without error message. | 
|  | </para> | 
|  | </listitem> | 
|  | <listitem> | 
|  | <para> | 
|  | Some useful programs: | 
|  | </para> | 
|  | <variablelist> | 
|  | <varlistentry> | 
|  | <term> | 
|  | <application>IDA</application>: | 
|  | <filename> | 
|  | <ulink url="ftp://ftp.uni-koeln.de/pc/msdos/programming/assembler/ida35bx.zip"> | 
|  | ftp://ftp.uni-koeln.de/pc/msdos/programming/assembler/ida35bx.zip</ulink> | 
|  | </filename> | 
|  | </term> | 
|  | <listitem> | 
|  | <para> | 
|  | *Very* good DOS disassembler ! It's badly needed | 
|  | for debugging Wine sometimes. | 
|  | </para> | 
|  | </listitem> | 
|  | </varlistentry> | 
|  | <varlistentry> | 
|  | <term> | 
|  | <application>XRAY</application>: | 
|  | <filename> | 
|  | <ulink url="ftp://ftp.th-darmstadt.de/pub/machines/ms-dos/SimTel/msdos/asmutil/xray15.zip"> | 
|  | ftp://ftp.th-darmstadt.de/pub/machines/ms-dos/SimTel/msdos/asmutil/xray15.zip</ulink> | 
|  | </filename> | 
|  | </term> | 
|  | <listitem> | 
|  | <para> | 
|  | Traces DOS calls (Int 21h, DPMI, ...). Use it with | 
|  | Windows to correct file management problems etc. | 
|  | </para> | 
|  | </listitem> | 
|  | </varlistentry> | 
|  | <varlistentry> | 
|  | <term> | 
|  | <application>pedump</application>: | 
|  | <filename> | 
|  | <ulink url="http://oak.oakland.edu/pub/simtelnet/win95/prog/pedump.zip"> | 
|  | http://oak.oakland.edu/pub/simtelnet/win95/prog/pedump.zip</ulink> | 
|  | </filename> | 
|  | </term> | 
|  | <listitem> | 
|  | <para> | 
|  | Dumps the imports and exports of a PE (Portable | 
|  | Executable) DLL. | 
|  | </para> | 
|  | </listitem> | 
|  | </varlistentry> | 
|  | </variablelist> | 
|  | </listitem> | 
|  | </itemizedlist> | 
|  | </sect2> | 
|  |  | 
|  | <sect2> | 
|  | <title>Some basic debugger usages:</title> | 
|  |  | 
|  | <para> | 
|  | After starting your program with | 
|  | </para> | 
|  | <screen> | 
|  | wine -debug myprog.exe | 
|  | </screen> | 
|  | <para> | 
|  | the program loads and you get a prompt at the program | 
|  | starting point. Then you can set breakpoints: | 
|  | </para> | 
|  | <screen> | 
|  | b RoutineName      (by outine name) OR | 
|  | b *0x812575        (by address) | 
|  | </screen> | 
|  | <para> | 
|  | Then you hit <command>c</command> (continue) to run the | 
|  | program. It stops at the breakpoint. You can type | 
|  | </para> | 
|  | <screen> | 
|  | step               (to step one line) OR | 
|  | stepi              (to step one machine instruction at a time; | 
|  | here, it helps to know the basic 386 | 
|  | instruction set) | 
|  | info reg           (to see registers) | 
|  | info stack         (to see hex values in the stack) | 
|  | info local         (to see local variables) | 
|  | list <line number> (to list source code) | 
|  | x <variable name>  (to examine a variable; only works if code | 
|  | is not compiled with optimization) | 
|  | x 0x4269978        (to examine a memory location) | 
|  | ?                  (help) | 
|  | q                  (quit) | 
|  | </screen> | 
|  | <para> | 
|  | By hitting <keycap>Enter</keycap>, you repeat the last | 
|  | command. | 
|  | </para> | 
|  | </sect2> | 
|  | </sect1> | 
|  |  | 
|  | <sect1 id="cvs-regression"> | 
|  | <title>How to do regression testing using Cvs</title> | 
|  |  | 
|  | <para> | 
|  | written by Gerard Patel (???) | 
|  | </para> | 
|  | <para> | 
|  | (Extracted from <filename>wine/documentation/bugreports</filename>) | 
|  | </para> | 
|  |  | 
|  | <para> | 
|  | A problem that can happen sometimes is 'it used to work | 
|  | before, now it doesn't anymore...'. Here is a step by step | 
|  | procedure to try to pinpoint when the problem occured. This is | 
|  | *NOT* for casual users. | 
|  | </para> | 
|  |  | 
|  | <orderedlist> | 
|  | <listitem> | 
|  | <para> | 
|  | Get the 'full cvs' archive from winehq. This archive is | 
|  | the cvs tree but with the tags controling the versioning | 
|  | system. It's a big file (> 15 meg) with a name like | 
|  | full-cvs-<last update date> (it's more than 100mb | 
|  | when uncompressed, you can't very well do this with | 
|  | small, old computers or slow Internet connections). | 
|  | </para> | 
|  | </listitem> | 
|  | <listitem> | 
|  | <para> | 
|  | untar it into a repository directory: | 
|  | <screen> | 
|  | cd /home/gerard | 
|  | tar -zxffull-cvs-2000-05-20.tar.gz | 
|  | mv wine repository | 
|  | </screen> | 
|  | </para> | 
|  | </listitem> | 
|  | <listitem> | 
|  | <para> | 
|  | extract a new destination directory. This directory must | 
|  | not be in a subdirectory of the repository else | 
|  | <command>cvs</command> will think it's part of the | 
|  | repository and deny you an extraction in the repository: | 
|  | <screen> | 
|  | cd /home/gerard | 
|  | mv wine wine_current (-> this protects your current wine sandbox, if any) | 
|  | export CVSROOT=/home/gerard/repository | 
|  | cd /home/gerard | 
|  | cvs -d $CVSROOT checkout wine | 
|  | </screen> | 
|  | </para> | 
|  | <para> | 
|  | Note that it's not possible to do a checkout at a given | 
|  | date; you always do the checkout for the last date where | 
|  | the full-cvs-xxx snapshot was generated. | 
|  | </para> | 
|  | </listitem> | 
|  | <listitem> | 
|  | <para> | 
|  | you will have now in the <filename>~/wine</filename> | 
|  | directory an image of the cvs tree, on the client side. | 
|  | Now update this image to the date you want: | 
|  | <screen> | 
|  | cd /home/gerard/wine | 
|  | cvs -d $CVSROOT update -D "1999-06-01" | 
|  | </screen> | 
|  | </para> | 
|  | <para> | 
|  | The date format is <literal>YYYY-MM-DD</literal>. | 
|  | </para> | 
|  | <para> | 
|  | Many messages will inform you that more recent files have | 
|  | been deleted to set back the client cvs tree to the date | 
|  | you asked, for example: | 
|  | <screen> | 
|  | cvs update: tsx11/ts_xf86dga2.c is no longer in the repository | 
|  | </screen> | 
|  | </para> | 
|  | <para> | 
|  | <command>cvs update</command> is not limited to upgrade to | 
|  | a *newer* version as I have believed for far too long :-( | 
|  | </para> | 
|  | </listitem> | 
|  | <listitem> | 
|  | <para> | 
|  | Now proceed as for a normal update: | 
|  | </para> | 
|  | <screen> | 
|  | ./configure | 
|  | make depend && make | 
|  | </screen> | 
|  | <para> | 
|  | When you have found the exact date when a bug was added to | 
|  | the cvs tree, use something like : | 
|  | <screen> | 
|  | cvs -d $CVSROOT diff -D "1999-07-10" -D "1999-07-12" | 
|  | </screen> | 
|  | to get all the differences between the last cvs tree | 
|  | version known to work and code that first displayed the | 
|  | misbehavior. | 
|  | </para> | 
|  | <note> | 
|  | <para> | 
|  | I did not include flags for <command>diff</command> | 
|  | since they are in my <filename>.cvsrc</filename> file: | 
|  | </para> | 
|  | <screen> | 
|  | cvs -z 3 | 
|  | update -dPA | 
|  | diff -u | 
|  | </screen> | 
|  | </note> | 
|  | <para> | 
|  | From this diff file, particularly the file names, and the | 
|  | <filename>ChangeLog</filename>, it's usually possible to | 
|  | find the different individual patches that were done at | 
|  | this time. | 
|  | </para> | 
|  | <para> | 
|  | If any non-programmer reads this, the fastest method to get | 
|  | at the point where the problem occured is to use a binary | 
|  | search, that is, if the problem occured in 1999, start at | 
|  | mid-year, then is the problem is already here, back to 1st | 
|  | April, if not, to 1st October, and so on. | 
|  | </para> | 
|  | </listitem> | 
|  | <listitem> | 
|  | <para> | 
|  | The next step is to start from the last working version | 
|  | and to dig the individual contributions from | 
|  | <ulink url="http://www.integrita.com/cgi-local/lwgate.pl/WINE-PATCHES/"> | 
|  | http://www.integrita.com/cgi-local/lwgate.pl/WINE-PATCHES/</ulink> | 
|  | (where the Wine patches mailing list is archived) | 
|  | </para> | 
|  | <para> | 
|  | If the patch was done by the Wine maintainer or if it was | 
|  | sent directly to his mail address without going first through | 
|  | <ulink url="mailto:wine-patches@winehq.com">wine-patches</ulink>, | 
|  | you are out of luck as you will never find the patch in | 
|  | the archive. If it is, it's often possible to apply the | 
|  | patches one by one to last working cvs snapshot, compile and test. | 
|  | If you have saved the next candidate as | 
|  | <filename>/home/gerard/buggedpatch1.txt</filename>: | 
|  | </para> | 
|  | <screen> | 
|  | cd /home/gerard/wine | 
|  | patch -p 0 < /home/gerard/buggedpatch1.txt | 
|  | </screen> | 
|  | <para> | 
|  | Beware that the committed patch is not always identical to | 
|  | the patch that the author sent to wine-patches, as | 
|  | sometimes the Wine maintainer changes things a bit. | 
|  | </para> | 
|  | <para> | 
|  | If you find one patch that is getting the cvs source tree to | 
|  | reproduce the problem, you have almost won; post the problem on | 
|  | <systemitem>comp.emulators.windows.wine</systemitem> and there | 
|  | is a chance that the author will jump in to suggest a fix; or | 
|  | there is always the possibility to look hard at the patch until | 
|  | it is coerced to reveal where is the bug :-) | 
|  | </para> | 
|  | </listitem> | 
|  | </orderedlist> | 
|  | </sect1> | 
|  | </chapter> | 
|  |  | 
|  | <!-- Keep this comment at the end of the file | 
|  | Local variables: | 
|  | mode: sgml | 
|  | sgml-parent-document:("wine-doc.sgml" "book" "part" "chapter" "") | 
|  | End: | 
|  | --> |