Index: uspace/srv/kbd/ctl/gxe_fb.c
===================================================================
--- uspace/srv/kbd/ctl/gxe_fb.c	(revision 12b679611efd5018a62a6d30b097bd4d7c89fe9c)
+++ uspace/srv/kbd/ctl/gxe_fb.c	(revision 9646159109db804ee73ff4a9fc8359fb3318c3d2)
@@ -48,5 +48,5 @@
 static void parse_ds_e1c(int scancode);
 
-static void parse_leaf(int scancode, int *map, size_t map_length);
+static void parse_leaf(int scancode, int (*map)[2], size_t map_length);
 
 enum dec_state {
@@ -59,115 +59,163 @@
 };
 
-static int map_start[] = {
-
-	[0x60] = KC_BACKTICK,
-
-	[0x31] = KC_1,
-	[0x32] = KC_2,
-	[0x33] = KC_3,
-	[0x34] = KC_4,
-	[0x35] = KC_5,
-	[0x36] = KC_6,
-	[0x37] = KC_7,
-	[0x38] = KC_8,
-	[0x39] = KC_9,
-	[0x30] = KC_0,
-
-	[0x2d] = KC_MINUS,
-	[0x3d] = KC_EQUALS,
-	[0x08] = KC_BACKSPACE,
-
-	[0x0f] = KC_TAB,
-
-	[0x71] = KC_Q,
-	[0x77] = KC_W,
-	[0x65] = KC_E,
-	[0x72] = KC_R,
-	[0x74] = KC_T,
-	[0x79] = KC_Y,
-	[0x75] = KC_U,
-	[0x69] = KC_I,
-	[0x6f] = KC_O,
-	[0x70] = KC_P,
-
-	[0x5b] = KC_LBRACKET,
-	[0x5d] = KC_RBRACKET,
-
-//	[0x3a] = KC_CAPS_LOCK,
-
-	[0x61] = KC_A,
-	[0x73] = KC_S,
-	[0x64] = KC_D,
-	[0x66] = KC_F,
-	[0x67] = KC_G,
-	[0x68] = KC_H,
-	[0x6a] = KC_J,
-	[0x6b] = KC_K,
-	[0x6c] = KC_L,
-
-	[0x3b] = KC_SEMICOLON,
-	[0x27] = KC_QUOTE,
-	[0x5c] = KC_BACKSLASH,
-
-//	[0x2a] = KC_LSHIFT,
-
-	[0x7a] = KC_Z,
-	[0x78] = KC_X,
-	[0x63] = KC_C,
-	[0x76] = KC_V,
-	[0x62] = KC_B,
-	[0x6e] = KC_N,
-	[0x6d] = KC_M,
-
-	[0x2c] = KC_COMMA,
-	[0x2e] = KC_PERIOD,
-	[0x2f] = KC_SLASH,
-
-//	[0x36] = KC_RSHIFT,
-
-//	[0x1d] = KC_LCTRL,
-//	[0x38] = KC_LALT,
-	[0x20] = KC_SPACE,
-
-	[0x1b] = KC_ESCAPE,
-
-	[0x0a] = KC_ENTER,
-	[0x0d] = KC_ENTER
-
-/*
-	[0x1] = KC_PRNSCR,
-	[0x1] = KC_SCROLL_LOCK,
-	[0x1] = KC_PAUSE,
-*/
-};
-
-static int map_e1[] =
-{
-};
-
-static int map_e1a[] =
-{
-	[0x50] = KC_F1,
-	[0x51] = KC_F2,
-	[0x52] = KC_F3,
-	[0x53] = KC_F4,
-};
-
-static int map_e1b[] =
-{
-	[0x33] = KC_F5,
-	[0x37] = KC_F6,
-	[0x38] = KC_F7,
-	[0x39] = KC_F8,
-};
-
-static int map_e1c[] =
-{
-	[0x38] = KC_F9,
-	[0x39] = KC_F10,
-	[0x33] = KC_F11,
-	[0x34] = KC_F12,
-};
-
+static int map_start[][2] = {
+
+	[0x60] = { 0, KC_BACKTICK },
+
+	[0x31] = { 0, KC_1 },
+	[0x32] = { 0, KC_2 },
+	[0x33] = { 0, KC_3 },
+	[0x34] = { 0, KC_4 },
+	[0x35] = { 0, KC_5 },
+	[0x36] = { 0, KC_6 },
+	[0x37] = { 0, KC_7 },
+	[0x38] = { 0, KC_8 },
+	[0x39] = { 0, KC_9 },
+	[0x30] = { 0, KC_0 },
+
+	[0x2d] = { 0, KC_MINUS },
+	[0x3d] = { 0, KC_EQUALS },
+	[0x08] = { 0, KC_BACKSPACE },
+
+	[0x0f] = { 0, KC_TAB },
+
+	[0x71] = { 0, KC_Q },
+	[0x77] = { 0, KC_W },
+	[0x65] = { 0, KC_E },
+	[0x72] = { 0, KC_R },
+	[0x74] = { 0, KC_T },
+	[0x79] = { 0, KC_Y },
+	[0x75] = { 0, KC_U },
+	[0x69] = { 0, KC_I },
+	[0x6f] = { 0, KC_O },
+	[0x70] = { 0, KC_P },
+
+	[0x5b] = { 0, KC_LBRACKET },
+	[0x5d] = { 0, KC_RBRACKET },
+
+	[0x61] = { 0, KC_A },
+	[0x73] = { 0, KC_S },
+	[0x64] = { 0, KC_D },
+	[0x66] = { 0, KC_F },
+	[0x67] = { 0, KC_G },
+	[0x68] = { 0, KC_H },
+	[0x6a] = { 0, KC_J },
+	[0x6b] = { 0, KC_K },
+	[0x6c] = { 0, KC_L },
+
+	[0x3b] = { 0, KC_SEMICOLON },
+	[0x27] = { 0, KC_QUOTE },
+	[0x5c] = { 0, KC_BACKSLASH },
+
+	[0x7a] = { 0, KC_Z },
+	[0x78] = { 0, KC_X },
+	[0x63] = { 0, KC_C },
+	[0x76] = { 0, KC_V },
+	[0x62] = { 0, KC_B },
+	[0x6e] = { 0, KC_N },
+	[0x6d] = { 0, KC_M },
+
+	[0x2c] = { 0, KC_COMMA },
+	[0x2e] = { 0, KC_PERIOD },
+	[0x2f] = { 0, KC_SLASH },
+
+	[0x20] = { 0, KC_SPACE },
+
+	[0x1b] = { 0, KC_ESCAPE },
+
+	[0x0a] = { 0, KC_ENTER },
+	[0x0d] = { 0, KC_ENTER },
+
+	/* with Shift pressed */
+
+	[0x7e] = { KM_LSHIFT, KC_BACKTICK },
+
+	[0x21] = { KM_LSHIFT, KC_1 },
+	[0x40] = { KM_LSHIFT, KC_2 },
+	[0x23] = { KM_LSHIFT, KC_3 },
+	[0x24] = { KM_LSHIFT, KC_4 },
+	[0x25] = { KM_LSHIFT, KC_5 },
+	[0x5e] = { KM_LSHIFT, KC_6 },
+	[0x26] = { KM_LSHIFT, KC_7 },
+	[0x2a] = { KM_LSHIFT, KC_8 },
+	[0x28] = { KM_LSHIFT, KC_9 },
+	[0x29] = { KM_LSHIFT, KC_0 },
+
+	[0x5f] = { KM_LSHIFT, KC_MINUS },
+	[0x2b] = { KM_LSHIFT, KC_EQUALS },
+
+	[0x51] = { KM_LSHIFT, KC_Q },
+	[0x57] = { KM_LSHIFT, KC_W },
+	[0x45] = { KM_LSHIFT, KC_E },
+	[0x52] = { KM_LSHIFT, KC_R },
+	[0x54] = { KM_LSHIFT, KC_T },
+	[0x59] = { KM_LSHIFT, KC_Y },
+	[0x55] = { KM_LSHIFT, KC_U },
+	[0x49] = { KM_LSHIFT, KC_I },
+	[0x4f] = { KM_LSHIFT, KC_O },
+	[0x50] = { KM_LSHIFT, KC_P },
+
+	[0x7b] = { KM_LSHIFT, KC_LBRACKET },
+	[0x7d] = { KM_LSHIFT, KC_RBRACKET },
+
+	[0x41] = { KM_LSHIFT, KC_A },
+	[0x53] = { KM_LSHIFT, KC_S },
+	[0x44] = { KM_LSHIFT, KC_D },
+	[0x46] = { KM_LSHIFT, KC_F },
+	[0x47] = { KM_LSHIFT, KC_G },
+	[0x48] = { KM_LSHIFT, KC_H },
+	[0x4a] = { KM_LSHIFT, KC_J },
+	[0x4b] = { KM_LSHIFT, KC_K },
+	[0x4c] = { KM_LSHIFT, KC_L },
+
+	[0x3a] = { KM_LSHIFT, KC_SEMICOLON },
+	[0x22] = { KM_LSHIFT, KC_QUOTE },
+	[0x7c] = { KM_LSHIFT, KC_BACKSLASH },
+
+	[0x5a] = { KM_LSHIFT, KC_Z },
+	[0x58] = { KM_LSHIFT, KC_X },
+	[0x43] = { KM_LSHIFT, KC_C },
+	[0x56] = { KM_LSHIFT, KC_V },
+	[0x42] = { KM_LSHIFT, KC_B },
+	[0x4e] = { KM_LSHIFT, KC_N },
+	[0x4d] = { KM_LSHIFT, KC_M },
+
+	[0x3c] = { KM_LSHIFT, KC_COMMA },
+	[0x3e] = { KM_LSHIFT, KC_PERIOD },
+	[0x3f] = { KM_LSHIFT, KC_SLASH }
+};
+
+static int map_e1[][2] =
+{
+};
+
+static int map_e1a[][2] =
+{
+	[0x50] = { 0, KC_F1 },
+	[0x51] = { 0, KC_F2 },
+	[0x52] = { 0, KC_F3 },
+	[0x53] = { 0, KC_F4 },
+};
+
+static int map_e1b[][2] =
+{
+	[0x33] = { 0, KC_F5 },
+	[0x37] = { 0, KC_F6 },
+	[0x38] = { 0, KC_F7 },
+	[0x39] = { 0, KC_F8 },
+};
+
+static int map_e1c[][2] =
+{
+	[0x38] = { 0, KC_F9 },
+	[0x39] = { 0, KC_F10 },
+	[0x33] = { 0, KC_F11 },
+	[0x34] = { 0, KC_F12 },
+};
+
+static unsigned int mods_keys[][2] = {
+	{ KM_LSHIFT, KC_LSHIFT },
+	{ 0, 0 }
+};
 
 static enum dec_state ds = ds_start;
@@ -192,5 +240,5 @@
 	}
 
-	parse_leaf(scancode, map_start, sizeof(map_start) / sizeof(int));
+	parse_leaf(scancode, map_start, sizeof(map_start) / (2 * sizeof(int)));
 }
 
@@ -215,25 +263,26 @@
 	}
 
-	parse_leaf(scancode, map_e1, sizeof(map_e1) / sizeof(int));
+	parse_leaf(scancode, map_e1, sizeof(map_e1) / (2 * sizeof(int)));
 }
 
 static void parse_ds_e1a(int scancode)
 {
-	parse_leaf(scancode, map_e1a, sizeof(map_e1a) / sizeof(int));
+	parse_leaf(scancode, map_e1a, sizeof(map_e1a) / (2 * sizeof(int)));
 }
 
 static void parse_ds_e1b(int scancode)
 {
-	parse_leaf(scancode, map_e1b, sizeof(map_e1b) / sizeof(int));
+	parse_leaf(scancode, map_e1b, sizeof(map_e1b) / (2 * sizeof(int)));
 }
 
 static void parse_ds_e1c(int scancode)
 {
-	parse_leaf(scancode, map_e1c, sizeof(map_e1c) / sizeof(int));
-}
-
-static void parse_leaf(int scancode, int *map, size_t map_length)
-{
-	unsigned int key;
+	parse_leaf(scancode, map_e1c, sizeof(map_e1c) / (2 * sizeof(int)));
+}
+
+static void parse_leaf(int scancode, int (*map)[2], size_t map_length)
+{
+	unsigned int key, mod;
+	int i;
 
 	ds = ds_start;
@@ -242,9 +291,30 @@
 		return;
 
-	key = map[scancode];
-	if (key != 0)
+	mod = map[scancode][0];
+	key = map[scancode][1];
+
+	/* Simulate modifier pressing. */
+	i = 0;
+	while (mods_keys[i][0] != 0) {
+		if (mod & mods_keys[i][0]) {
+			kbd_push_ev(KE_PRESS, mods_keys[i][1]);
+		}
+		++i;
+	}
+
+	if (key != 0) {
 		kbd_push_ev(KE_PRESS, key);
-}
-
+		kbd_push_ev(KE_RELEASE, key);
+	}
+
+	/* Simulate modifier releasing. */
+	i = 0;
+	while (mods_keys[i][0] != 0) {
+		if (mod & mods_keys[i][0]) {
+			kbd_push_ev(KE_RELEASE, mods_keys[i][1]);
+		}
+		++i;
+	}
+}
 
 /**
Index: uspace/srv/kbd/ctl/stty.c
===================================================================
--- uspace/srv/kbd/ctl/stty.c	(revision 12b679611efd5018a62a6d30b097bd4d7c89fe9c)
+++ uspace/srv/kbd/ctl/stty.c	(revision 9646159109db804ee73ff4a9fc8359fb3318c3d2)
@@ -48,5 +48,5 @@
 static void parse_ds_e2b(int scancode);
 
-static void parse_leaf(int scancode, int *map, size_t map_length);
+static void parse_leaf(int scancode, int (*map)[2], size_t map_length);
 
 enum dec_state {
@@ -59,119 +59,167 @@
 };
 
-static int map_start[] = {
-
-	[0x60] = KC_BACKTICK,
-
-	[0x31] = KC_1,
-	[0x32] = KC_2,
-	[0x33] = KC_3,
-	[0x34] = KC_4,
-	[0x35] = KC_5,
-	[0x36] = KC_6,
-	[0x37] = KC_7,
-	[0x38] = KC_8,
-	[0x39] = KC_9,
-	[0x30] = KC_0,
-
-	[0x2d] = KC_MINUS,
-	[0x3d] = KC_EQUALS,
-	[0x08] = KC_BACKSPACE,
-
-	[0x0f] = KC_TAB,
-
-	[0x71] = KC_Q,
-	[0x77] = KC_W,
-	[0x65] = KC_E,
-	[0x72] = KC_R,
-	[0x74] = KC_T,
-	[0x79] = KC_Y,
-	[0x75] = KC_U,
-	[0x69] = KC_I,
-	[0x6f] = KC_O,
-	[0x70] = KC_P,
-
-	[0x5b] = KC_LBRACKET,
-	[0x5d] = KC_RBRACKET,
-
-//	[0x3a] = KC_CAPS_LOCK,
-
-	[0x61] = KC_A,
-	[0x73] = KC_S,
-	[0x64] = KC_D,
-	[0x66] = KC_F,
-	[0x67] = KC_G,
-	[0x68] = KC_H,
-	[0x6a] = KC_J,
-	[0x6b] = KC_K,
-	[0x6c] = KC_L,
-
-	[0x3b] = KC_SEMICOLON,
-	[0x27] = KC_QUOTE,
-	[0x5c] = KC_BACKSLASH,
-
-//	[0x2a] = KC_LSHIFT,
-
-	[0x7a] = KC_Z,
-	[0x78] = KC_X,
-	[0x63] = KC_C,
-	[0x76] = KC_V,
-	[0x62] = KC_B,
-	[0x6e] = KC_N,
-	[0x6d] = KC_M,
-
-	[0x2c] = KC_COMMA,
-	[0x2e] = KC_PERIOD,
-	[0x2f] = KC_SLASH,
-
-//	[0x36] = KC_RSHIFT,
-
-//	[0x1d] = KC_LCTRL,
-//	[0x38] = KC_LALT,
-	[0x20] = KC_SPACE,
-
-	[0x1b] = KC_ESCAPE,
-
-	[0x0a] = KC_ENTER,
-	[0x0d] = KC_ENTER
-
-/*
-	[0x1] = KC_PRNSCR,
-	[0x1] = KC_SCROLL_LOCK,
-	[0x1] = KC_PAUSE,
-*/
-};
-
-static int map_e1[] =
-{
-	[0x50] = KC_F1,
-	[0x51] = KC_F2,
-	[0x52] = KC_F3,
-	[0x53] = KC_F4,
-};
-
-static int map_e2[] =
-{
-	[0x41] = KC_UP,
-	[0x42] = KC_DOWN,
-	[0x44] = KC_LEFT,
-	[0x43] = KC_RIGHT,
-};
-
-static int map_e2a[] =
-{
-	[0x35] = KC_F5,
-	[0x37] = KC_F6,
-	[0x38] = KC_F7,
-	[0x39] = KC_F8,
-};
-
-static int map_e2b[] =
-{
-	[0x30] = KC_F9,
-	[0x31] = KC_F10,
-	[0x32] = KC_F11,
-	[0x33] = KC_F12,
-};
-
+static int map_start[][2] = {
+
+	[0x60] = { 0, KC_BACKTICK },
+
+	[0x31] = { 0, KC_1 },
+	[0x32] = { 0, KC_2 },
+	[0x33] = { 0, KC_3 },
+	[0x34] = { 0, KC_4 },
+	[0x35] = { 0, KC_5 },
+	[0x36] = { 0, KC_6 },
+	[0x37] = { 0, KC_7 },
+	[0x38] = { 0, KC_8 },
+	[0x39] = { 0, KC_9 },
+	[0x30] = { 0, KC_0 },
+
+	[0x2d] = { 0, KC_MINUS },
+	[0x3d] = { 0, KC_EQUALS },
+	[0x08] = { 0, KC_BACKSPACE },
+
+	[0x0f] = { 0, KC_TAB },
+
+	[0x71] = { 0, KC_Q },
+	[0x77] = { 0, KC_W },
+	[0x65] = { 0, KC_E },
+	[0x72] = { 0, KC_R },
+	[0x74] = { 0, KC_T },
+	[0x79] = { 0, KC_Y },
+	[0x75] = { 0, KC_U },
+	[0x69] = { 0, KC_I },
+	[0x6f] = { 0, KC_O },
+	[0x70] = { 0, KC_P },
+
+	[0x5b] = { 0, KC_LBRACKET },
+	[0x5d] = { 0, KC_RBRACKET },
+
+	[0x61] = { 0, KC_A },
+	[0x73] = { 0, KC_S },
+	[0x64] = { 0, KC_D },
+	[0x66] = { 0, KC_F },
+	[0x67] = { 0, KC_G },
+	[0x68] = { 0, KC_H },
+	[0x6a] = { 0, KC_J },
+	[0x6b] = { 0, KC_K },
+	[0x6c] = { 0, KC_L },
+
+	[0x3b] = { 0, KC_SEMICOLON },
+	[0x27] = { 0, KC_QUOTE },
+	[0x5c] = { 0, KC_BACKSLASH },
+
+	[0x7a] = { 0, KC_Z },
+	[0x78] = { 0, KC_X },
+	[0x63] = { 0, KC_C },
+	[0x76] = { 0, KC_V },
+	[0x62] = { 0, KC_B },
+	[0x6e] = { 0, KC_N },
+	[0x6d] = { 0, KC_M },
+
+	[0x2c] = { 0, KC_COMMA },
+	[0x2e] = { 0, KC_PERIOD },
+	[0x2f] = { 0, KC_SLASH },
+
+	[0x20] = { 0, KC_SPACE },
+
+	[0x1b] = { 0, KC_ESCAPE },
+
+	[0x0a] = { 0, KC_ENTER },
+	[0x0d] = { 0, KC_ENTER },
+
+	/* with Shift pressed */
+
+	[0x7e] = { KM_LSHIFT, KC_BACKTICK },
+
+	[0x21] = { KM_LSHIFT, KC_1 },
+	[0x40] = { KM_LSHIFT, KC_2 },
+	[0x23] = { KM_LSHIFT, KC_3 },
+	[0x24] = { KM_LSHIFT, KC_4 },
+	[0x25] = { KM_LSHIFT, KC_5 },
+	[0x5e] = { KM_LSHIFT, KC_6 },
+	[0x26] = { KM_LSHIFT, KC_7 },
+	[0x2a] = { KM_LSHIFT, KC_8 },
+	[0x28] = { KM_LSHIFT, KC_9 },
+	[0x29] = { KM_LSHIFT, KC_0 },
+
+	[0x5f] = { KM_LSHIFT, KC_MINUS },
+	[0x2b] = { KM_LSHIFT, KC_EQUALS },
+
+	[0x51] = { KM_LSHIFT, KC_Q },
+	[0x57] = { KM_LSHIFT, KC_W },
+	[0x45] = { KM_LSHIFT, KC_E },
+	[0x52] = { KM_LSHIFT, KC_R },
+	[0x54] = { KM_LSHIFT, KC_T },
+	[0x59] = { KM_LSHIFT, KC_Y },
+	[0x55] = { KM_LSHIFT, KC_U },
+	[0x49] = { KM_LSHIFT, KC_I },
+	[0x4f] = { KM_LSHIFT, KC_O },
+	[0x50] = { KM_LSHIFT, KC_P },
+
+	[0x7b] = { KM_LSHIFT, KC_LBRACKET },
+	[0x7d] = { KM_LSHIFT, KC_RBRACKET },
+
+	[0x41] = { KM_LSHIFT, KC_A },
+	[0x53] = { KM_LSHIFT, KC_S },
+	[0x44] = { KM_LSHIFT, KC_D },
+	[0x46] = { KM_LSHIFT, KC_F },
+	[0x47] = { KM_LSHIFT, KC_G },
+	[0x48] = { KM_LSHIFT, KC_H },
+	[0x4a] = { KM_LSHIFT, KC_J },
+	[0x4b] = { KM_LSHIFT, KC_K },
+	[0x4c] = { KM_LSHIFT, KC_L },
+
+	[0x3a] = { KM_LSHIFT, KC_SEMICOLON },
+	[0x22] = { KM_LSHIFT, KC_QUOTE },
+	[0x7c] = { KM_LSHIFT, KC_BACKSLASH },
+
+	[0x5a] = { KM_LSHIFT, KC_Z },
+	[0x58] = { KM_LSHIFT, KC_X },
+	[0x43] = { KM_LSHIFT, KC_C },
+	[0x56] = { KM_LSHIFT, KC_V },
+	[0x42] = { KM_LSHIFT, KC_B },
+	[0x4e] = { KM_LSHIFT, KC_N },
+	[0x4d] = { KM_LSHIFT, KC_M },
+
+	[0x3c] = { KM_LSHIFT, KC_COMMA },
+	[0x3e] = { KM_LSHIFT, KC_PERIOD },
+	[0x3f] = { KM_LSHIFT, KC_SLASH }
+};
+
+static int map_e1[][2] =
+{
+	[0x50] = { 0, KC_F1 },
+	[0x51] = { 0, KC_F2 },
+	[0x52] = { 0, KC_F3 },
+	[0x53] = { 0, KC_F4 },
+};
+
+static int map_e2[][2] =
+{
+	[0x41] = { 0, KC_UP },
+	[0x42] = { 0, KC_DOWN },
+	[0x44] = { 0, KC_LEFT },
+	[0x43] = { 0, KC_RIGHT },
+};
+
+static int map_e2a[][2] =
+{
+	[0x35] = { 0, KC_F5 },
+	[0x37] = { 0, KC_F6 },
+	[0x38] = { 0, KC_F7 },
+	[0x39] = { 0, KC_F8 },
+};
+
+static int map_e2b[][2] =
+{
+	[0x30] = { 0, KC_F9 },
+	[0x31] = { 0, KC_F10 },
+	[0x32] = { 0, KC_F11 },
+	[0x33] = { 0, KC_F12 },
+};
+
+static unsigned int mods_keys[][2] = {
+	{ KM_LSHIFT, KC_LSHIFT },
+	{ 0, 0 }
+};
 
 static enum dec_state ds;
@@ -191,7 +239,4 @@
 static void parse_ds_start(int scancode)
 {
-	if (scancode < 0 || scancode >= sizeof(map_start) / sizeof(int))
-		return;
-
 	if (scancode == 0x1b) {
 		ds = ds_e;
@@ -199,5 +244,5 @@
 	}
 
-	parse_leaf(scancode, map_start, sizeof(map_start) / sizeof(int));
+	parse_leaf(scancode, map_start, sizeof(map_start) / (2 * sizeof(int)));
 }
 
@@ -218,5 +263,5 @@
 static void parse_ds_e1(int scancode)
 {
-	parse_leaf(scancode, map_e1, sizeof(map_e1) / sizeof(int));
+	parse_leaf(scancode, map_e1, sizeof(map_e1) / (2 * sizeof(int)));
 }
 
@@ -229,20 +274,21 @@
 	}
 
-	parse_leaf(scancode, map_e2, sizeof(map_e2) / sizeof(int));
+	parse_leaf(scancode, map_e2, sizeof(map_e2) / (2 * sizeof(int)));
 }
 
 static void parse_ds_e2a(int scancode)
 {
-	parse_leaf(scancode, map_e2a, sizeof(map_e2a) / sizeof(int));
+	parse_leaf(scancode, map_e2a, sizeof(map_e2a) / (2 * sizeof(int)));
 }
 
 static void parse_ds_e2b(int scancode)
 {
-	parse_leaf(scancode, map_e2b, sizeof(map_e2b) / sizeof(int));
-}
-
-static void parse_leaf(int scancode, int *map, size_t map_length)
-{
-	unsigned int key;
+	parse_leaf(scancode, map_e2b, sizeof(map_e2b) / (2 * sizeof(int)));
+}
+
+static void parse_leaf(int scancode, int (*map)[2], size_t map_length)
+{
+	unsigned int key, mod;
+	int i;
 
 	ds = ds_start;
@@ -251,7 +297,29 @@
 		return;
 
-	key = map[scancode];
-	if (key != 0)
+	mod = map[scancode][0];
+	key = map[scancode][1];
+
+	/* Simulate modifier pressing. */
+	i = 0;
+	while (mods_keys[i][0] != 0) {
+		if (mod & mods_keys[i][0]) {
+			kbd_push_ev(KE_PRESS, mods_keys[i][1]);
+		}
+		++i;
+	}
+
+	if (key != 0) {
 		kbd_push_ev(KE_PRESS, key);
+		kbd_push_ev(KE_RELEASE, key);
+	}
+
+	/* Simulate modifier releasing. */
+	i = 0;
+	while (mods_keys[i][0] != 0) {
+		if (mod & mods_keys[i][0]) {
+			kbd_push_ev(KE_RELEASE, mods_keys[i][1]);
+		}
+		++i;
+	}
 }
 
Index: uspace/srv/kbd/generic/kbd.c
===================================================================
--- uspace/srv/kbd/generic/kbd.c	(revision 12b679611efd5018a62a6d30b097bd4d7c89fe9c)
+++ uspace/srv/kbd/generic/kbd.c	(revision 9646159109db804ee73ff4a9fc8359fb3318c3d2)
@@ -69,5 +69,5 @@
 void kbd_push_scancode(int scancode)
 {
-	printf("scancode: 0x%x\n", scancode);
+/*	printf("scancode: 0x%x\n", scancode);*/
 	kbd_ctl_parse_scancode(scancode);
 }
@@ -115,9 +115,9 @@
 		}
 	}
-
+/*
 	printf("type: %d\n", type);
 	printf("mods: 0x%x\n", mods);
 	printf("keycode: %u\n", key);
-
+*/
 	ev.type = type;
 	ev.key = key;
