class.c 7.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262
  1. /*
  2. * Server-side window class management
  3. *
  4. * Copyright (C) 2002 Mike McCormack
  5. * Copyright (C) 2003 Alexandre Julliard
  6. *
  7. * This library is free software; you can redistribute it and/or
  8. * modify it under the terms of the GNU Lesser General Public
  9. * License as published by the Free Software Foundation; either
  10. * version 2.1 of the License, or (at your option) any later version.
  11. *
  12. * This library is distributed in the hope that it will be useful,
  13. * but WITHOUT ANY WARRANTY; without even the implied warranty of
  14. * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
  15. * Lesser General Public License for more details.
  16. *
  17. * You should have received a copy of the GNU Lesser General Public
  18. * License along with this library; if not, write to the Free Software
  19. * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
  20. */
  21. #include "config.h"
  22. #include "wine/port.h"
  23. #include <assert.h>
  24. #include <stdlib.h>
  25. #include <stdio.h>
  26. #include <string.h>
  27. #include "wine/list.h"
  28. #include "request.h"
  29. #include "object.h"
  30. #include "process.h"
  31. #include "user.h"
  32. struct window_class
  33. {
  34. struct list entry; /* entry in process list */
  35. struct process *process; /* process owning the class */
  36. int count; /* reference count */
  37. int local; /* local class? */
  38. atom_t atom; /* class atom */
  39. void *instance; /* module instance */
  40. unsigned int style; /* class style */
  41. int win_extra; /* number of window extra bytes */
  42. void *client_ptr; /* pointer to class in client address space */
  43. int nb_extra_bytes; /* number of extra bytes */
  44. char extra_bytes[1]; /* extra bytes storage */
  45. };
  46. #define DESKTOP_ATOM ((atom_t)32769)
  47. static struct window_class *desktop_class;
  48. static struct window_class *create_class( struct process *process, int extra_bytes, int local )
  49. {
  50. struct window_class *class;
  51. if (!(class = mem_alloc( sizeof(*class) + extra_bytes - 1 ))) return NULL;
  52. class->process = (struct process *)grab_object( process );
  53. class->count = 0;
  54. class->local = local;
  55. class->nb_extra_bytes = extra_bytes;
  56. memset( class->extra_bytes, 0, extra_bytes );
  57. /* other fields are initialized by caller */
  58. /* local classes have priority so we put them first in the list */
  59. if (local) list_add_head( &process->classes, &class->entry );
  60. else list_add_tail( &process->classes, &class->entry );
  61. return class;
  62. }
  63. static struct window_class *create_desktop_class( unsigned int style, int win_extra )
  64. {
  65. struct window_class *class;
  66. if (!(class = mem_alloc( sizeof(*class) - 1 ))) return NULL;
  67. class->process = NULL;
  68. class->count = 0;
  69. class->local = 0;
  70. class->nb_extra_bytes = 0;
  71. class->atom = DESKTOP_ATOM;
  72. class->instance = NULL;
  73. class->style = style;
  74. class->win_extra = win_extra;
  75. class->client_ptr = NULL;
  76. desktop_class = class;
  77. return class;
  78. }
  79. static void destroy_class( struct window_class *class )
  80. {
  81. list_remove( &class->entry );
  82. release_object( class->process );
  83. free( class );
  84. }
  85. void destroy_process_classes( struct process *process )
  86. {
  87. struct list *ptr;
  88. while ((ptr = list_head( &process->classes )))
  89. {
  90. struct window_class *class = LIST_ENTRY( ptr, struct window_class, entry );
  91. destroy_class( class );
  92. }
  93. }
  94. static struct window_class *find_class( struct process *process, atom_t atom, void *instance )
  95. {
  96. struct list *ptr;
  97. LIST_FOR_EACH( ptr, &process->classes )
  98. {
  99. struct window_class *class = LIST_ENTRY( ptr, struct window_class, entry );
  100. if (class->atom != atom) continue;
  101. if (!instance || !class->local || class->instance == instance) return class;
  102. }
  103. if (atom == DESKTOP_ATOM) return desktop_class;
  104. return NULL;
  105. }
  106. struct window_class *grab_class( struct process *process, atom_t atom,
  107. void *instance, int *extra_bytes )
  108. {
  109. struct window_class *class = find_class( process, atom, instance );
  110. if (class)
  111. {
  112. class->count++;
  113. *extra_bytes = class->win_extra;
  114. }
  115. else set_error( STATUS_INVALID_HANDLE );
  116. return class;
  117. }
  118. void release_class( struct window_class *class )
  119. {
  120. assert( class->count > 0 );
  121. class->count--;
  122. }
  123. atom_t get_class_atom( struct window_class *class )
  124. {
  125. return class->atom;
  126. }
  127. void *get_class_client_ptr( struct window_class *class )
  128. {
  129. return class->client_ptr;
  130. }
  131. /* create a window class */
  132. DECL_HANDLER(create_class)
  133. {
  134. struct window_class *class;
  135. if (!req->local && req->atom == DESKTOP_ATOM)
  136. {
  137. if (!desktop_class) create_desktop_class( req->style, req->win_extra );
  138. return; /* silently ignore further attempts to create the desktop class */
  139. }
  140. class = find_class( current->process, req->atom, req->instance );
  141. if (class && !class->local == !req->local)
  142. {
  143. set_win32_error( ERROR_CLASS_ALREADY_EXISTS );
  144. return;
  145. }
  146. if (req->extra < 0 || req->extra > 4096 || req->win_extra < 0 || req->win_extra > 4096)
  147. {
  148. /* don't allow stupid values here */
  149. set_error( STATUS_INVALID_PARAMETER );
  150. return;
  151. }
  152. if (!grab_global_atom( req->atom )) return;
  153. if (!(class = create_class( current->process, req->extra, req->local )))
  154. {
  155. release_global_atom( req->atom );
  156. return;
  157. }
  158. class->atom = req->atom;
  159. class->instance = req->instance;
  160. class->style = req->style;
  161. class->win_extra = req->win_extra;
  162. class->client_ptr = req->client_ptr;
  163. }
  164. /* destroy a window class */
  165. DECL_HANDLER(destroy_class)
  166. {
  167. struct window_class *class = find_class( current->process, req->atom, req->instance );
  168. if (!class)
  169. set_win32_error( ERROR_CLASS_DOES_NOT_EXIST );
  170. else if (class->count)
  171. set_win32_error( ERROR_CLASS_HAS_WINDOWS );
  172. else
  173. {
  174. reply->client_ptr = class->client_ptr;
  175. if (class != desktop_class) destroy_class( class );
  176. }
  177. }
  178. /* set some information in a class */
  179. DECL_HANDLER(set_class_info)
  180. {
  181. struct window_class *class = get_window_class( req->window );
  182. if (!class) return;
  183. if (req->flags && class->process != current->process)
  184. {
  185. set_error( STATUS_ACCESS_DENIED );
  186. return;
  187. }
  188. if (req->extra_size > sizeof(req->extra_value) ||
  189. req->extra_offset < -1 ||
  190. req->extra_offset > class->nb_extra_bytes - (int)req->extra_size)
  191. {
  192. set_win32_error( ERROR_INVALID_INDEX );
  193. return;
  194. }
  195. if ((req->flags & SET_CLASS_WINEXTRA) && (req->win_extra < 0 || req->win_extra > 4096))
  196. {
  197. set_error( STATUS_INVALID_PARAMETER );
  198. return;
  199. }
  200. if (req->extra_offset != -1)
  201. {
  202. memcpy( &reply->old_extra_value, class->extra_bytes + req->extra_offset, req->extra_size );
  203. }
  204. else if (req->flags & SET_CLASS_EXTRA)
  205. {
  206. set_win32_error( ERROR_INVALID_INDEX );
  207. return;
  208. }
  209. reply->old_atom = class->atom;
  210. reply->old_style = class->style;
  211. reply->old_extra = class->nb_extra_bytes;
  212. reply->old_win_extra = class->win_extra;
  213. reply->old_instance = class->instance;
  214. if (req->flags & SET_CLASS_ATOM)
  215. {
  216. if (!grab_global_atom( req->atom )) return;
  217. release_global_atom( class->atom );
  218. class->atom = req->atom;
  219. }
  220. if (req->flags & SET_CLASS_STYLE) class->style = req->style;
  221. if (req->flags & SET_CLASS_WINEXTRA) class->win_extra = req->win_extra;
  222. if (req->flags & SET_CLASS_INSTANCE) class->instance = req->instance;
  223. if (req->flags & SET_CLASS_EXTRA) memcpy( class->extra_bytes + req->extra_offset,
  224. &req->extra_value, req->extra_size );
  225. }