[CMAKE]
[reactos.git] / dll / keyboard / kbdfr / kbdfr.c
1 /*
2 * ReactOS French keyboard layout
3 * Copyright (C) 2003 ReactOS
4 * License: LGPL, see: LGPL.txt
5 * Autors: Jean-Michel Gay, 2003
6 * Pierre Schweitzer, 2007
7 *
8 */
9
10 #include <windows.h>
11 #include <internal/kbd.h>
12
13 #ifdef _M_IA64
14 #define ROSDATA static __declspec(allocate(".data"))
15 #else
16 #ifdef _MSC_VER
17 #pragma data_seg(".data")
18 #define ROSDATA static
19 #else
20 #define ROSDATA static __attribute__((section(".data")))
21 #endif
22 #endif
23
24 #define VK_EMPTY 0xff /* The non-existent VK */
25 #define KSHIFT 0x001 /* Shift modifier */
26 #define KCTRL 0x002 /* Ctrl modifier */
27 #define KALT 0x004 /* Alt modifier */
28 #define KEXT 0x100 /* Extended key code */
29 #define KMULTI 0x200 /* Multi-key */
30 #define KSPEC 0x400 /* Special key */
31 #define KNUMP 0x800 /* Number-pad */
32 #define KNUMS 0xc00 /* Special + number pad */
33 #define KMEXT 0x300 /* Multi + ext */
34
35 #define SHFT_INVALID 0x0F
36
37 /**
38 * FIXME : - VK_DIVIDE produces ! instead of /
39 * - VK_SNAPSHOT produces *
40 */
41
42 ROSDATA USHORT scancode_to_vk[] = {
43 /* Numbers Row */
44 /* - 00 - */
45 /* 1 ... 2 ... 3 ... 4 ... */
46 VK_EMPTY, VK_ESCAPE, '1', '2',
47 '3', '4', '5', '6',
48 '7', '8', '9', '0',
49 VK_OEM_4, VK_OEM_PLUS, VK_BACK,
50 /* - 0f - */
51 /* First Letters Row */
52 VK_TAB, 'A', 'Z', 'E',
53 'R', 'T', 'Y', 'U',
54 'I', 'O', 'P',
55 VK_OEM_6, VK_OEM_1, VK_RETURN,
56 /* - 1d - */
57 /* Second Letters Row */
58 VK_LCONTROL,
59 'Q', 'S', 'D', 'F',
60 'G', 'H', 'J', 'K',
61 'L', 'M' , VK_OEM_3, VK_OEM_7,
62 VK_LSHIFT, VK_OEM_5,
63 /* - 2c - */
64 /* Third letters row */
65 'W', 'X', 'C', 'V',
66 'B', 'N', VK_OEM_COMMA,
67 VK_OEM_PERIOD,VK_OEM_2, VK_OEM_8, VK_RSHIFT,
68 /* - 37 - */
69 /* Bottom Row */
70 VK_MULTIPLY, VK_LMENU, VK_SPACE, VK_CAPITAL,
71
72 /* - 3b - */
73 /* F-Keys */
74 VK_F1, VK_F2, VK_F3, VK_F4, VK_F5, VK_F6,
75 VK_F7, VK_F8, VK_F9, VK_F10,
76 /* - 45 - */
77 /* Locks */
78 VK_NUMLOCK | KMEXT,
79 VK_SCROLL | KMULTI,
80 /* - 47 - */
81 /* Number-Pad */
82 VK_HOME | KNUMS, VK_UP | KNUMS, VK_PRIOR | KNUMS, VK_SUBTRACT,
83 VK_LEFT | KNUMS, VK_CLEAR | KNUMS, VK_RIGHT | KNUMS, VK_ADD,
84 VK_END | KNUMS, VK_DOWN | KNUMS, VK_NEXT | KNUMS,
85 VK_INSERT | KNUMS, VK_DELETE | KNUMS,
86 /* - 54 - */
87 /* Presumably PrtSc */
88 VK_SNAPSHOT,
89 /* - 55 - */
90 /* Oddities, and the remaining standard F-Keys */
91 VK_EMPTY, VK_OEM_102, VK_F11, VK_F12,
92 /* - 59 - */
93 VK_CLEAR, VK_EMPTY, VK_EMPTY, VK_EMPTY, VK_EMPTY, /* EREOF */
94 VK_EMPTY, VK_EMPTY, VK_EMPTY, VK_EMPTY, VK_ZOOM, /* ZOOM */
95 VK_HELP,
96 /* - 64 - */
97 /* Even more F-Keys (for example, NCR keyboards from the early 90's) */
98 VK_F13, VK_F14, VK_F15, VK_F16, VK_F17, VK_F18, VK_F19, VK_F20,
99 VK_F21, VK_F22, VK_F23,
100 /* - 6f - */
101 /* Not sure who uses these codes */
102 VK_EMPTY, VK_EMPTY, VK_EMPTY,
103 /* - 72 - */
104 VK_EMPTY, VK_EMPTY, VK_EMPTY, VK_EMPTY,
105 /* - 76 - */
106 /* One more f-key */
107 VK_F24,
108 /* - 77 - */
109 VK_EMPTY, VK_EMPTY, VK_EMPTY, VK_EMPTY,
110 VK_EMPTY, VK_EMPTY, VK_EMPTY, VK_EMPTY, /* PA1 */
111 VK_EMPTY,
112 /* - 80 - */
113 0
114 };
115
116 ROSDATA VSC_VK extcode0_to_vk[] = {
117 { 0x10, VK_MEDIA_PREV_TRACK | KEXT },
118 { 0x19, VK_MEDIA_NEXT_TRACK | KEXT },
119 { 0x1D, VK_RCONTROL | KEXT },
120 { 0x20, VK_VOLUME_MUTE | KEXT },
121 { 0x21, VK_LAUNCH_APP2 | KEXT },
122 { 0x22, VK_MEDIA_PLAY_PAUSE | KEXT },
123 { 0x24, VK_MEDIA_STOP | KEXT },
124 { 0x2E, VK_VOLUME_DOWN | KEXT },
125 { 0x30, VK_VOLUME_UP | KEXT },
126 { 0x32, VK_BROWSER_HOME | KEXT },
127 { 0x35, VK_DIVIDE | KEXT },
128 { 0x37, VK_SNAPSHOT | KEXT },
129 { 0x38, VK_RMENU | KEXT },
130 { 0x47, VK_HOME | KEXT },
131 { 0x48, VK_UP | KEXT },
132 { 0x49, VK_PRIOR | KEXT },
133 { 0x4B, VK_LEFT | KEXT },
134 { 0x4D, VK_RIGHT | KEXT },
135 { 0x4F, VK_END | KEXT },
136 { 0x50, VK_DOWN | KEXT },
137 { 0x51, VK_NEXT | KEXT },
138 { 0x52, VK_INSERT | KEXT },
139 { 0x53, VK_DELETE | KEXT },
140 { 0x5B, VK_LWIN | KEXT },
141 { 0x5C, VK_RWIN | KEXT },
142 { 0x5D, VK_APPS | KEXT },
143 { 0x5F, VK_SLEEP | KEXT },
144 { 0x65, VK_BROWSER_SEARCH | KEXT },
145 { 0x66, VK_BROWSER_FAVORITES | KEXT },
146 { 0x67, VK_BROWSER_REFRESH | KEXT },
147 { 0x68, VK_BROWSER_STOP | KEXT },
148 { 0x69, VK_BROWSER_FORWARD | KEXT },
149 { 0x6A, VK_BROWSER_BACK | KEXT },
150 { 0x6B, VK_LAUNCH_APP1 | KEXT },
151 { 0x6C, VK_LAUNCH_MAIL | KEXT },
152 { 0x6D, VK_LAUNCH_MEDIA_SELECT | KEXT },
153 { 0x1C, VK_RETURN | KEXT },
154 { 0x46, VK_CANCEL | KEXT },
155 { 0, 0 },
156 };
157
158 ROSDATA VSC_VK extcode1_to_vk[] = {
159 { 0, 0 },
160 };
161
162 ROSDATA VK_TO_BIT modifier_keys[] = {
163 { VK_SHIFT, KSHIFT },
164 { VK_CONTROL, KCTRL },
165 { VK_MENU, KALT },
166 { 0, 0 }
167 };
168
169 ROSDATA MODIFIERS modifier_bits = {
170 modifier_keys,
171 6,
172 { 0, 1, 2, 4, SHFT_INVALID, SHFT_INVALID, 3 }
173 /* Modifier bit order: NONE, SHIFT, CTRL, ALT, ?, ?, SHIFT-CTRL */
174 };
175
176 #define NOCAPS 0
177 #define CAPS KSHIFT /* Caps -> shift */
178
179 ROSDATA VK_TO_WCHARS2 key_to_chars_2mod[] = {
180 /* Normal vs Shifted */
181 /* The numbers */
182 { '1', NOCAPS, {'&', '1'} },
183 /* Specials */
184 /* Ctrl-_ generates FR */
185 { VK_OEM_5 ,NOCAPS, {'*',0xb5} }, /* \9c */
186 { VK_OEM_3 ,NOCAPS, {0xf9, '%'} }, /* ù */
187 { VK_OEM_COMMA ,NOCAPS, {',', '?'} },
188 { VK_OEM_PERIOD ,NOCAPS, {';', '.'} },
189 { VK_OEM_2 ,NOCAPS, {':', '/'} },
190 { VK_OEM_8 ,NOCAPS, {'!', 0xa7} }, /* § */
191 /* Keys that do not have shift states */
192 { VK_TAB ,NOCAPS, {'\t','\t'} },
193 { VK_ADD ,NOCAPS, {'+', '+'} },
194 { VK_SUBTRACT ,NOCAPS, {'-', '-'} },
195 { VK_MULTIPLY ,NOCAPS, {'*', '*'} },
196 { VK_DIVIDE ,NOCAPS, {'/', '/'} },
197 { VK_ESCAPE ,NOCAPS, {'\x1b','\x1b'} },
198 { VK_SPACE ,NOCAPS, {' ', ' '} },
199 { 0, 0 }
200 };
201
202 ROSDATA VK_TO_WCHARS3 key_to_chars_3mod[] = {
203 /* Normal, Shifted, Ctrl */
204
205 /* The alphabet */
206 { 'A', CAPS, {'a', 'A', 0x01} },
207 { 'B', CAPS, {'b', 'B', 0x02} },
208 { 'C', CAPS, {'c', 'C', 0x03} },
209 { 'D', CAPS, {'d', 'D', 0x04} },
210 { 'F', CAPS, {'f', 'F', 0x06} },
211 { 'G', CAPS, {'g', 'G', 0x07} },
212 { 'H', CAPS, {'h', 'H', 0x08} },
213 { 'I', CAPS, {'i', 'I', 0x09} },
214 { 'J', CAPS, {'j', 'J', 0x0a} },
215 { 'K', CAPS, {'k', 'K', 0x0b} },
216 { 'L', CAPS, {'l', 'L', 0x0c} },
217 { 'M', CAPS, {'m', 'M', 0x0d} },
218 { 'N', CAPS, {'n', 'N', 0x0e} },
219 { 'O', CAPS, {'o', 'O', 0x0f} },
220 { 'P', CAPS, {'p', 'P', 0x10} },
221 { 'Q', CAPS, {'q', 'Q', 0x11} },
222 { 'R', CAPS, {'r', 'R', 0x12} },
223 { 'S', CAPS, {'s', 'S', 0x13} },
224 { 'T', CAPS, {'t', 'T', 0x14} },
225 { 'U', CAPS, {'u', 'U', 0x15} },
226 { 'V', CAPS, {'v', 'V', 0x16} },
227 { 'W', CAPS, {'w', 'W', 0x17} },
228 { 'X', CAPS, {'x', 'X', 0x18} },
229 { 'Y', CAPS, {'y', 'Y', 0x19} },
230 { 'Z', CAPS, {'z', 'Z', 0x1a} },
231
232 /* Legacy (telnet-style) ascii escapes */
233 { VK_OEM_102, 0, {'<', '>', 0x1c /* FS */} },
234 { VK_OEM_6, 0, {WCH_DEAD, WCH_DEAD, WCH_NONE} },
235 { VK_EMPTY, 0, {'^', 0xa8, WCH_NONE} }, //OEM 6 DEAD ¨
236 { VK_OEM_7, 0, {0xb2,0xb3, 0x1c /* FS */} }, /* ², ³ */
237 { VK_RETURN, 0, {'\r', '\r', '\n'} },
238 { 0,0 }
239 };
240
241 ROSDATA VK_TO_WCHARS4 key_to_chars_4mod[] = {
242 /* Normal, shifted, control, Alt+Gr */
243 { '2' , CAPS, {0xe9, '2', WCH_NONE, WCH_DEAD} },
244 { VK_EMPTY, NOCAPS, {WCH_NONE, WCH_NONE, WCH_NONE, '~'} },
245 { '3' , NOCAPS, {'"', '3', WCH_NONE, '#'} },
246 { '4' , NOCAPS, {'\'', '4', WCH_NONE, '{'} },
247 { '7' , CAPS, {0xe8, '7', WCH_NONE, WCH_DEAD} },
248 { VK_EMPTY, NOCAPS, {WCH_NONE, WCH_NONE, WCH_NONE, '`'} },
249 { '9' , CAPS, {0xe7, '9', WCH_NONE, '^'} }, /* ç */
250 { '0' , CAPS, {0xe0, '0', WCH_NONE, '@'} }, /* à */
251 { VK_OEM_PLUS,NOCAPS, {'=', '+', WCH_NONE, '}'} },
252 { 'E' , CAPS, {'e', 'E', 0x05, 0x20ac } }, /* \80 */
253 { VK_OEM_1, NOCAPS, {0x24, 0xa3, WCH_NONE, 0xa4} }, /* $, £, ¤ */
254 { VK_OEM_4, NOCAPS, {')', 0xb0, WCH_NONE, ']'} }, /* ° */
255 { 0, 0 }
256 };
257
258 ROSDATA VK_TO_WCHARS5 key_to_chars_5mod[] = {
259 /* x,x, Normal, Shifted, Ctrl, Alt, C-S-x */
260 /* Legacy Ascii generators */
261 { '5' , 1 , {'(', '5', WCH_NONE , '[' , 0x1b} },
262 { '6' , 1 , {'-', '6', WCH_NONE , '|' , 0x1f} },
263 { '8' , 1 , {'_', '8', WCH_NONE , '\\' , 0x1c} },
264 { 0, 0 }
265 };
266
267 ROSDATA VK_TO_WCHARS1 keypad_numbers[] = {
268 { VK_NUMPAD0, 0, {'0'} },
269 { VK_NUMPAD1, 0, {'1'} },
270 { VK_NUMPAD2, 0, {'2'} },
271 { VK_NUMPAD3, 0, {'3'} },
272 { VK_NUMPAD4, 0, {'4'} },
273 { VK_NUMPAD5, 0, {'5'} },
274 { VK_NUMPAD6, 0, {'6'} },
275 { VK_NUMPAD7, 0, {'7'} },
276 { VK_NUMPAD8, 0, {'8'} },
277 { VK_NUMPAD9, 0, {'9'} },
278 { VK_DECIMAL, 0, {'.'} },
279 { VK_BACK, 0, {'\010'} },
280 { 0,0 }
281 };
282
283 #define vk_master(n,x) { (PVK_TO_WCHARS1)x, n, sizeof(x[0]) }
284
285 ROSDATA VK_TO_WCHAR_TABLE vk_to_wchar_master_table[] = {
286 vk_master(1,keypad_numbers),
287 vk_master(2,key_to_chars_2mod),
288 vk_master(3,key_to_chars_3mod),
289 vk_master(4,key_to_chars_4mod),
290 vk_master(5,key_to_chars_5mod),
291 { 0,0,0 }
292 };
293
294 #define DEADTRANS(ch, accent, comp, flags) MAKELONG(ch, accent), comp, flags
295 ROSDATA DEADKEY deadkey[] =
296 {
297 //*´*
298 { DEADTRANS(L'a', 0xb4, 0xe1, 0x00) }, // á
299 { DEADTRANS(L'A', 0xb4, 0xc1, 0x00) }, // Á
300 { DEADTRANS(L'e', 0xb4, 0xe9, 0x00) }, // é
301 { DEADTRANS(L'E', 0xb4, 0xc9, 0x00) }, // É
302 { DEADTRANS(L'i', 0xb4, 0xeD, 0x00) }, // í
303 { DEADTRANS(L'I', 0xb4, 0xcD, 0x00) }, // Í
304 { DEADTRANS(L'o', 0xb4, 0xf3, 0x00) }, // ó
305 { DEADTRANS(L'O', 0xb4, 0xd3, 0x00) }, // Ó
306 { DEADTRANS(L'u', 0xb4, 0xfa, 0x00) }, // ú
307 { DEADTRANS(L'U', 0xb4, 0xda, 0x00) }, // Ú
308 { DEADTRANS(L'y', 0xb4, 0xfd, 0x00) }, // ý
309 { DEADTRANS(L'Y', 0xb4, 0xdd, 0x00) }, // Ý
310 //*`*
311 { DEADTRANS(L'a', 0x60, 0xe0, 0x00) }, // à
312 { DEADTRANS(L'A', 0x60, 0xc0, 0x00) }, // À
313 { DEADTRANS(L'e', 0x60, 0xe8, 0x00) }, // è
314 { DEADTRANS(L'E', 0x60, 0xc8, 0x00) }, // È
315 { DEADTRANS(L'i', 0x60, 0xec, 0x00) }, // ì
316 { DEADTRANS(L'I', 0x60, 0xcc, 0x00) }, // Ì
317 { DEADTRANS(L'o', 0x60, 0xf2, 0x00) }, // ò
318 { DEADTRANS(L'O', 0x60, 0xd2, 0x00) }, // Ò
319 { DEADTRANS(L'u', 0x60, 0xf9, 0x00) }, // ù
320 { DEADTRANS(L'U', 0x60, 0xd9, 0x00) }, // Ù
321 //*^*
322 { DEADTRANS(L'a', 0x5e, 0xe2, 0x00) }, // â
323 { DEADTRANS(L'A', 0x5e, 0xd2, 0x00) }, // Â
324 { DEADTRANS(L'e', 0x5e, 0xea, 0x00) }, // ê
325 { DEADTRANS(L'E', 0x5e, 0xca, 0x00) }, // Ê
326 { DEADTRANS(L'i', 0x5e, 0xee, 0x00) }, // î
327 { DEADTRANS(L'I', 0x5e, 0xce, 0x00) }, // Î
328 { DEADTRANS(L'o', 0x5e, 0xf4, 0x00) }, // ô
329 { DEADTRANS(L'O', 0x5e, 0xd4, 0x00) }, // Ô
330 { DEADTRANS(L'u', 0x5e, 0xfb, 0x00) }, // û
331 { DEADTRANS(L'U', 0x5e, 0xdb, 0x00) }, // Û
332 //*¨*
333 { DEADTRANS(L'a', 0xa8, 0xe4, 0x00) }, // ä
334 { DEADTRANS(L'A', 0xa8, 0xc4, 0x00) }, // Ä
335 { DEADTRANS(L'e', 0xa8, 0xeb, 0x00) }, // ë
336 { DEADTRANS(L'E', 0xa8, 0xcb, 0x00) }, // Ë
337 { DEADTRANS(L'i', 0xa8, 0xef, 0x00) }, // ï
338 { DEADTRANS(L'I', 0xa8, 0xcf, 0x00) }, // Ï
339 { DEADTRANS(L'o', 0xa8, 0xf6, 0x00) }, // ö
340 { DEADTRANS(L'O', 0xa8, 0xd6, 0x00) }, // Ö
341 { DEADTRANS(L'u', 0xa8, 0xfc, 0x00) }, // ü
342 { DEADTRANS(L'U', 0xa8, 0xdc, 0x00) }, // Ü
343 { DEADTRANS(L'y', 0xa8, 0xff, 0x00) }, // ÿ
344 //*~*
345 { DEADTRANS(L'a', 0x7e, 0xe3, 0x00) }, // ã
346 { DEADTRANS(L'A', 0x7e, 0xc3, 0x00) }, // Ã
347 { DEADTRANS(L'n', 0x7e, 0xf1, 0x00) }, // ñ
348 { DEADTRANS(L'N', 0x7e, 0xd1, 0x00) }, // Ñ
349 { DEADTRANS(L'o', 0x7e, 0xf5, 0x00) }, // õ
350 { DEADTRANS(L'O', 0x7e, 0xd5, 0x00) }, // Õ
351 { 0, 0, 0}
352 };
353
354
355 ROSDATA VSC_LPWSTR key_names[] = {
356 { 0x00, L"" },
357 { 0x01, L"Esc" },
358 { 0x0e, L"Backspace" },
359 { 0x0f, L"Tab" },
360 { 0x1c, L"Enter" },
361 { 0x1d, L"Ctrl" },
362 { 0x2a, L"Shift" },
363 { 0x36, L"Right Shift" },
364 { 0x37, L"Num *" },
365 { 0x38, L"Alt" },
366 { 0x39, L"Space" },
367 { 0x3a, L"Caps Lock" },
368 { 0x3b, L"F1" },
369 { 0x3c, L"F2" },
370 { 0x3d, L"F3" },
371 { 0x3e, L"F4" },
372 { 0x3f, L"F5" },
373 { 0x40, L"F6" },
374 { 0x41, L"F7" },
375 { 0x42, L"F8" },
376 { 0x43, L"F9" },
377 { 0x44, L"F10" },
378 { 0x45, L"Pause" },
379 { 0x46, L"Scroll Lock" },
380 { 0x47, L"Num 7" },
381 { 0x48, L"Num 8" },
382 { 0x49, L"Num 9" },
383 { 0x4a, L"Num -" },
384 { 0x4b, L"Num 4" },
385 { 0x4c, L"Num 5" },
386 { 0x4d, L"Num 6" },
387 { 0x4e, L"Num +" },
388 { 0x4f, L"Num 1" },
389 { 0x50, L"Num 2" },
390 { 0x51, L"Num 3" },
391 { 0x52, L"Num 0" },
392 { 0x53, L"Num Del" },
393 { 0x54, L"Sys Req" },
394 { 0x57, L"F11" },
395 { 0x58, L"F12" },
396 { 0x7c, L"F13" },
397 { 0x7d, L"F14" },
398 { 0x7e, L"F15" },
399 { 0x7f, L"F16" },
400 { 0x80, L"F17" },
401 { 0x81, L"F18" },
402 { 0x82, L"F19" },
403 { 0x83, L"F20" },
404 { 0x84, L"F21" },
405 { 0x85, L"F22" },
406 { 0x86, L"F23" },
407 { 0x87, L"F24" },
408 { 0, NULL },
409 };
410
411 ROSDATA VSC_LPWSTR extended_key_names[] = {
412 { 0x1c, L"Num Enter" },
413 { 0x1d, L"Right Ctrl" },
414 { 0x35, L"Num /" },
415 { 0x37, L"Prnt Scrn" },
416 { 0x38, L"Right Alt" },
417 { 0x45, L"Num Lock" },
418 { 0x46, L"Break" },
419 { 0x47, L"Home" },
420 { 0x48, L"Up" },
421 { 0x49, L"Page Up" },
422 { 0x4a, L"Left" },
423 { 0x4c, L"Center" },
424 { 0x4d, L"Right" },
425 { 0x4f, L"End" },
426 { 0x50, L"Down" },
427 { 0x51, L"Page Down" },
428 { 0x52, L"Insert" },
429 { 0x53, L"Delete" },
430 { 0x54, L"<ReactOS>" },
431 { 0x55, L"Help" },
432 { 0x56, L"Left Windows" },
433 { 0x5b, L"Right Windows" },
434 { 0, NULL },
435 };
436
437 ROSDATA DEADKEY_LPWSTR dead_key_names[] = {
438 L"\x005e" L"Circumflex",
439 L"\x007e" L"Tilde",
440 L"\x0060" L"Grave",
441 NULL
442 };
443
444 /* Finally, the master table */
445 ROSDATA KBDTABLES keyboard_layout_table = {
446 /* modifier assignments */
447 &modifier_bits,
448
449 /* character from vk tables */
450 vk_to_wchar_master_table,
451
452 /* diacritical marks -- currently implemented by wine code */
453 deadkey,
454
455 /* Key names */
456 (VSC_LPWSTR *)key_names,
457 (VSC_LPWSTR *)extended_key_names,
458 dead_key_names, /* Dead key names */
459
460 /* scan code to virtual key maps */
461 scancode_to_vk,
462 sizeof(scancode_to_vk) / sizeof(scancode_to_vk[0]),
463 extcode0_to_vk,
464 extcode1_to_vk,
465
466 MAKELONG(1,1), /* Version 1.0 */
467
468 /* Ligatures -- French doesn't have any */
469 0,
470 0,
471 NULL
472 };
473
474 PKBDTABLES WINAPI KbdLayerDescriptor(VOID) {
475 return &keyboard_layout_table;
476 }
477