filename | src/drivers/video_gtk.c |
changeset | 669:ab344e42bca9 |
prev | 665:99ae9dc4cab7 |
next | 677:3ee62740ff8f |
author | nkeynes |
date | Mon May 12 10:00:13 2008 +0000 (15 years ago) |
permissions | -rw-r--r-- |
last change | Cleanup most of the -Wall warnings (getting a bit sloppy...) Convert FP code to use fixed banks rather than indirect pointer (3-4% faster this way now) |
file | annotate | diff | log | raw |
nkeynes@94 | 1 | /** |
nkeynes@586 | 2 | * $Id$ |
nkeynes@94 | 3 | * |
nkeynes@94 | 4 | * The PC side of the video support (responsible for actually displaying / |
nkeynes@94 | 5 | * rendering frames) |
nkeynes@94 | 6 | * |
nkeynes@94 | 7 | * Copyright (c) 2005 Nathan Keynes. |
nkeynes@94 | 8 | * |
nkeynes@94 | 9 | * This program is free software; you can redistribute it and/or modify |
nkeynes@94 | 10 | * it under the terms of the GNU General Public License as published by |
nkeynes@94 | 11 | * the Free Software Foundation; either version 2 of the License, or |
nkeynes@94 | 12 | * (at your option) any later version. |
nkeynes@94 | 13 | * |
nkeynes@94 | 14 | * This program is distributed in the hope that it will be useful, |
nkeynes@94 | 15 | * but WITHOUT ANY WARRANTY; without even the implied warranty of |
nkeynes@94 | 16 | * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the |
nkeynes@94 | 17 | * GNU General Public License for more details. |
nkeynes@94 | 18 | */ |
nkeynes@94 | 19 | |
nkeynes@485 | 20 | #include <gdk/gdkkeysyms.h> |
nkeynes@94 | 21 | #include <stdint.h> |
nkeynes@658 | 22 | #include "lxdream.h" |
nkeynes@144 | 23 | #include "display.h" |
nkeynes@608 | 24 | #include "dckeysyms.h" |
nkeynes@653 | 25 | #include "drivers/video_gl.h" |
nkeynes@669 | 26 | #include "drivers/joy_linux.h" |
nkeynes@586 | 27 | #include "pvr2/pvr2.h" |
nkeynes@537 | 28 | #include "gtkui/gtkui.h" |
nkeynes@94 | 29 | |
nkeynes@658 | 30 | #ifdef HAVE_GTK_X11 |
nkeynes@658 | 31 | |
nkeynes@658 | 32 | #include <gdk/gdkx.h> |
nkeynes@658 | 33 | #include "drivers/video_glx.h" |
nkeynes@658 | 34 | |
nkeynes@658 | 35 | /************* X11-specificness **********/ |
nkeynes@658 | 36 | |
nkeynes@658 | 37 | guint gdk_keycode_to_modifier( GdkDisplay *display, guint keycode ) |
nkeynes@658 | 38 | { |
nkeynes@658 | 39 | int i; |
nkeynes@658 | 40 | int result = 0; |
nkeynes@658 | 41 | Display *xdisplay = GDK_DISPLAY_XDISPLAY (display); |
nkeynes@658 | 42 | XModifierKeymap *keymap = XGetModifierMapping( xdisplay ); |
nkeynes@658 | 43 | for( i=0; i<8*keymap->max_keypermod; i++ ) { |
nkeynes@658 | 44 | if( keymap->modifiermap[i] == keycode ) { |
nkeynes@658 | 45 | result = 1 << (i/keymap->max_keypermod); |
nkeynes@658 | 46 | break; |
nkeynes@658 | 47 | } |
nkeynes@658 | 48 | } |
nkeynes@658 | 49 | XFreeModifiermap(keymap); |
nkeynes@658 | 50 | return result; |
nkeynes@658 | 51 | } |
nkeynes@658 | 52 | |
nkeynes@658 | 53 | #if !(GTK_CHECK_VERSION(2,8,0)) |
nkeynes@658 | 54 | /* gdk_display_warp_pointer was added in GTK 2.8. If we're using an earlier |
nkeynes@658 | 55 | * version, include the code here. (Can't just set the dependency on 2.8 as |
nkeynes@658 | 56 | * it still hasn't been included in fink yet...) Original copyright statement |
nkeynes@658 | 57 | * below. |
nkeynes@658 | 58 | */ |
nkeynes@658 | 59 | |
nkeynes@658 | 60 | /* GDK - The GIMP Drawing Kit |
nkeynes@658 | 61 | * Copyright (C) 1995-1997 Peter Mattis, Spencer Kimball and Josh MacDonald |
nkeynes@658 | 62 | * |
nkeynes@658 | 63 | * This library is free software; you can redistribute it and/or |
nkeynes@658 | 64 | * modify it under the terms of the GNU Lesser General Public |
nkeynes@658 | 65 | * License as published by the Free Software Foundation; either |
nkeynes@658 | 66 | * version 2 of the License, or (at your option) any later version. |
nkeynes@658 | 67 | * |
nkeynes@658 | 68 | * This library is distributed in the hope that it will be useful, |
nkeynes@658 | 69 | * but WITHOUT ANY WARRANTY; without even the implied warranty of |
nkeynes@658 | 70 | * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU |
nkeynes@658 | 71 | * Lesser General Public License for more details. |
nkeynes@658 | 72 | * |
nkeynes@658 | 73 | * You should have received a copy of the GNU Lesser General Public |
nkeynes@658 | 74 | * License along with this library; if not, write to the |
nkeynes@658 | 75 | * Free Software Foundation, Inc., 59 Temple Place - Suite 330, |
nkeynes@658 | 76 | * Boston, MA 02111-1307, USA. |
nkeynes@658 | 77 | */ |
nkeynes@658 | 78 | |
nkeynes@658 | 79 | /* |
nkeynes@658 | 80 | * Modified by the GTK+ Team and others 1997-2000. See the AUTHORS |
nkeynes@658 | 81 | * file for a list of people on the GTK+ Team. See the ChangeLog |
nkeynes@658 | 82 | * files for a list of changes. These files are distributed with |
nkeynes@658 | 83 | * GTK+ at ftp://ftp.gtk.org/pub/gtk/. |
nkeynes@658 | 84 | */ |
nkeynes@658 | 85 | void gdk_display_warp_pointer (GdkDisplay *display, |
nkeynes@658 | 86 | GdkScreen *screen, |
nkeynes@658 | 87 | gint x, |
nkeynes@658 | 88 | gint y) |
nkeynes@658 | 89 | { |
nkeynes@658 | 90 | Display *xdisplay; |
nkeynes@658 | 91 | Window dest; |
nkeynes@658 | 92 | |
nkeynes@658 | 93 | xdisplay = GDK_DISPLAY_XDISPLAY (display); |
nkeynes@658 | 94 | dest = GDK_WINDOW_XWINDOW (gdk_screen_get_root_window (screen)); |
nkeynes@658 | 95 | |
nkeynes@658 | 96 | XWarpPointer (xdisplay, None, dest, 0, 0, 0, 0, x, y); |
nkeynes@658 | 97 | } |
nkeynes@658 | 98 | |
nkeynes@658 | 99 | #endif |
nkeynes@658 | 100 | |
nkeynes@658 | 101 | #endif |
nkeynes@659 | 102 | |
nkeynes@663 | 103 | #ifdef HAVE_GTK_OSX |
nkeynes@663 | 104 | #include "drivers/video_nsgl.h" |
nkeynes@663 | 105 | |
nkeynes@663 | 106 | // Include this prototype as some systems don't have gdkquartz.h installed |
nkeynes@663 | 107 | NSView *gdk_quartz_window_get_nsview( GdkWindow *window); |
nkeynes@663 | 108 | |
nkeynes@663 | 109 | guint gdk_keycode_to_modifier( GdkDisplay *display, guint keycode ) |
nkeynes@663 | 110 | { |
nkeynes@663 | 111 | return 0; |
nkeynes@663 | 112 | } |
nkeynes@663 | 113 | |
nkeynes@663 | 114 | #endif |
nkeynes@663 | 115 | |
nkeynes@663 | 116 | |
nkeynes@663 | 117 | |
nkeynes@661 | 118 | GtkWidget *gtk_video_drawable = NULL; |
nkeynes@436 | 119 | int video_width = 640; |
nkeynes@436 | 120 | int video_height = 480; |
nkeynes@94 | 121 | |
nkeynes@352 | 122 | gboolean video_gtk_init(); |
nkeynes@352 | 123 | void video_gtk_shutdown(); |
nkeynes@669 | 124 | void video_gtk_display_blank( uint32_t colour ); |
nkeynes@144 | 125 | uint16_t video_gtk_resolve_keysym( const gchar *keysym ); |
nkeynes@614 | 126 | uint16_t video_gtk_keycode_to_dckeysym(uint16_t keycode); |
nkeynes@94 | 127 | |
nkeynes@370 | 128 | struct display_driver display_gtk_driver = { "gtk", video_gtk_init, video_gtk_shutdown, |
nkeynes@352 | 129 | video_gtk_resolve_keysym, |
nkeynes@608 | 130 | video_gtk_keycode_to_dckeysym, |
nkeynes@614 | 131 | NULL, |
nkeynes@552 | 132 | NULL, NULL, NULL, NULL, NULL, |
nkeynes@552 | 133 | video_gtk_display_blank, NULL }; |
nkeynes@449 | 134 | |
nkeynes@144 | 135 | uint16_t video_gtk_resolve_keysym( const gchar *keysym ) |
nkeynes@144 | 136 | { |
nkeynes@144 | 137 | int val = gdk_keyval_from_name( keysym ); |
nkeynes@144 | 138 | if( val == GDK_VoidSymbol ) |
nkeynes@144 | 139 | return 0; |
nkeynes@144 | 140 | return (uint16_t)val; |
nkeynes@144 | 141 | } |
nkeynes@144 | 142 | |
nkeynes@435 | 143 | gboolean video_gtk_expose_callback(GtkWidget *widget, GdkEventExpose *event, gpointer data ) |
nkeynes@435 | 144 | { |
nkeynes@545 | 145 | render_buffer_t buffer = pvr2_get_front_buffer(); |
nkeynes@545 | 146 | if( buffer == NULL ) { |
nkeynes@545 | 147 | display_gtk_driver.display_blank(pvr2_get_border_colour()); |
nkeynes@545 | 148 | } else { |
nkeynes@545 | 149 | display_gtk_driver.display_render_buffer(buffer); |
nkeynes@545 | 150 | } |
nkeynes@443 | 151 | return TRUE; |
nkeynes@435 | 152 | } |
nkeynes@435 | 153 | |
nkeynes@436 | 154 | gboolean video_gtk_resize_callback(GtkWidget *widget, GdkEventConfigure *event, gpointer data ) |
nkeynes@436 | 155 | { |
nkeynes@436 | 156 | video_width = event->width; |
nkeynes@436 | 157 | video_height = event->height; |
nkeynes@545 | 158 | video_gtk_expose_callback(widget, NULL, data); |
nkeynes@443 | 159 | return TRUE; |
nkeynes@436 | 160 | } |
nkeynes@436 | 161 | |
nkeynes@614 | 162 | uint16_t video_gtk_keycode_to_dckeysym(uint16_t keycode) |
nkeynes@608 | 163 | { |
nkeynes@608 | 164 | if( keycode >= 'a' && keycode <= 'z' ) { |
nkeynes@608 | 165 | return (keycode - 'a') + DCKB_A; |
nkeynes@608 | 166 | } else if( keycode >= '1' && keycode <= '9' ) { |
nkeynes@608 | 167 | return (keycode - '1') + DCKB_1; |
nkeynes@608 | 168 | } |
nkeynes@608 | 169 | switch(keycode) { |
nkeynes@609 | 170 | case GDK_0: return DCKB_0; |
nkeynes@609 | 171 | case GDK_Return: return DCKB_ENTER; |
nkeynes@609 | 172 | case GDK_Escape: return DCKB_ESCAPE; |
nkeynes@609 | 173 | case GDK_BackSpace: return DCKB_BACKSPACE; |
nkeynes@609 | 174 | case GDK_Tab: return DCKB_TAB; |
nkeynes@609 | 175 | case GDK_space: return DCKB_SPACE; |
nkeynes@609 | 176 | case GDK_minus: return DCKB_MINUS; |
nkeynes@609 | 177 | case GDK_equal: return DCKB_EQUAL; |
nkeynes@609 | 178 | case GDK_bracketleft: return DCKB_LBRACKET; |
nkeynes@609 | 179 | case GDK_bracketright: return DCKB_RBRACKET; |
nkeynes@609 | 180 | case GDK_semicolon: return DCKB_SEMICOLON; |
nkeynes@609 | 181 | case GDK_apostrophe:return DCKB_QUOTE; |
nkeynes@609 | 182 | case GDK_grave : return DCKB_BACKQUOTE; |
nkeynes@609 | 183 | case GDK_comma: return DCKB_COMMA; |
nkeynes@609 | 184 | case GDK_period: return DCKB_PERIOD; |
nkeynes@609 | 185 | case GDK_slash: return DCKB_SLASH; |
nkeynes@609 | 186 | case GDK_Caps_Lock: return DCKB_CAPSLOCK; |
nkeynes@609 | 187 | case GDK_F1: return DCKB_F1; |
nkeynes@609 | 188 | case GDK_F2: return DCKB_F2; |
nkeynes@609 | 189 | case GDK_F3: return DCKB_F3; |
nkeynes@609 | 190 | case GDK_F4: return DCKB_F4; |
nkeynes@609 | 191 | case GDK_F5: return DCKB_F5; |
nkeynes@609 | 192 | case GDK_F6: return DCKB_F6; |
nkeynes@609 | 193 | case GDK_F7: return DCKB_F7; |
nkeynes@609 | 194 | case GDK_F8: return DCKB_F8; |
nkeynes@609 | 195 | case GDK_F9: return DCKB_F9; |
nkeynes@609 | 196 | case GDK_F10: return DCKB_F10; |
nkeynes@609 | 197 | case GDK_F11: return DCKB_F11; |
nkeynes@609 | 198 | case GDK_F12: return DCKB_F12; |
nkeynes@609 | 199 | case GDK_Scroll_Lock: return DCKB_SCROLLLOCK; |
nkeynes@609 | 200 | case GDK_Pause: return DCKB_PAUSE; |
nkeynes@609 | 201 | case GDK_Insert: return DCKB_INSERT; |
nkeynes@609 | 202 | case GDK_Home: return DCKB_HOME; |
nkeynes@609 | 203 | case GDK_Page_Up: return DCKB_PAGEUP; |
nkeynes@609 | 204 | case GDK_Delete: return DCKB_DELETE; |
nkeynes@609 | 205 | case GDK_End: return DCKB_END; |
nkeynes@609 | 206 | case GDK_Page_Down: return DCKB_PAGEDOWN; |
nkeynes@609 | 207 | case GDK_Right: return DCKB_RIGHT; |
nkeynes@609 | 208 | case GDK_Left: return DCKB_LEFT; |
nkeynes@609 | 209 | case GDK_Down: return DCKB_DOWN; |
nkeynes@609 | 210 | case GDK_Up: return DCKB_UP; |
nkeynes@609 | 211 | case GDK_Num_Lock: return DCKB_NUMLOCK; |
nkeynes@609 | 212 | case GDK_KP_Divide: return DCKB_KP_SLASH; |
nkeynes@609 | 213 | case GDK_KP_Multiply: return DCKB_KP_STAR; |
nkeynes@609 | 214 | case GDK_KP_Subtract: return DCKB_KP_MINUS; |
nkeynes@609 | 215 | case GDK_KP_Add: return DCKB_KP_PLUS; |
nkeynes@609 | 216 | case GDK_KP_Enter: return DCKB_KP_ENTER; |
nkeynes@631 | 217 | case GDK_KP_End: return DCKB_KP_1; |
nkeynes@631 | 218 | case GDK_KP_Down: return DCKB_KP_2; |
nkeynes@631 | 219 | case GDK_KP_Page_Down: return DCKB_KP_3; |
nkeynes@631 | 220 | case GDK_KP_Left: return DCKB_KP_4; |
nkeynes@631 | 221 | case GDK_KP_Begin: return DCKB_KP_5; |
nkeynes@631 | 222 | case GDK_KP_Right: return DCKB_KP_6; |
nkeynes@631 | 223 | case GDK_KP_Home: return DCKB_KP_7; |
nkeynes@631 | 224 | case GDK_KP_Up: return DCKB_KP_8; |
nkeynes@631 | 225 | case GDK_KP_Page_Up:return DCKB_KP_9; |
nkeynes@631 | 226 | case GDK_KP_Insert: return DCKB_KP_0; |
nkeynes@632 | 227 | case GDK_KP_Delete: return DCKB_KP_PERIOD; |
nkeynes@609 | 228 | case GDK_backslash: return DCKB_BACKSLASH; |
nkeynes@609 | 229 | case GDK_Control_L: return DCKB_CONTROL_L; |
nkeynes@609 | 230 | case GDK_Shift_L: return DCKB_SHIFT_L; |
nkeynes@609 | 231 | case GDK_Alt_L: return DCKB_ALT_L; |
nkeynes@609 | 232 | case GDK_Meta_L: return DCKB_S1; |
nkeynes@609 | 233 | case GDK_Control_R: return DCKB_CONTROL_R; |
nkeynes@609 | 234 | case GDK_Shift_R: return DCKB_SHIFT_R; |
nkeynes@609 | 235 | case GDK_Alt_R: return DCKB_ALT_R; |
nkeynes@609 | 236 | case GDK_Meta_R: return DCKB_S2; |
nkeynes@608 | 237 | } |
nkeynes@608 | 238 | return DCKB_NONE; |
nkeynes@608 | 239 | } |
nkeynes@608 | 240 | |
nkeynes@659 | 241 | GtkWidget *video_gtk_create_drawable() |
nkeynes@659 | 242 | { |
nkeynes@659 | 243 | GtkWidget *drawable = gtk_drawing_area_new(); |
nkeynes@659 | 244 | GTK_WIDGET_SET_FLAGS(drawable, GTK_CAN_FOCUS|GTK_CAN_DEFAULT); |
nkeynes@659 | 245 | |
nkeynes@659 | 246 | g_signal_connect( drawable, "expose_event", |
nkeynes@659 | 247 | G_CALLBACK(video_gtk_expose_callback), NULL ); |
nkeynes@659 | 248 | g_signal_connect( drawable, "configure_event", |
nkeynes@659 | 249 | G_CALLBACK(video_gtk_resize_callback), NULL ); |
nkeynes@659 | 250 | |
nkeynes@659 | 251 | #ifdef HAVE_GLX |
nkeynes@659 | 252 | Display *display = gdk_x11_display_get_xdisplay( gtk_widget_get_display(drawable)); |
nkeynes@659 | 253 | Screen *screen = gdk_x11_screen_get_xscreen( gtk_widget_get_screen(drawable)); |
nkeynes@659 | 254 | int screen_no = XScreenNumberOfScreen(screen); |
nkeynes@659 | 255 | if( !video_glx_init(display, screen_no) ) { |
nkeynes@659 | 256 | ERROR( "Unable to initialize GLX, aborting" ); |
nkeynes@659 | 257 | exit(3); |
nkeynes@659 | 258 | } |
nkeynes@659 | 259 | |
nkeynes@659 | 260 | XVisualInfo *visual = video_glx_get_visual(); |
nkeynes@659 | 261 | if( visual != NULL ) { |
nkeynes@659 | 262 | GdkVisual *gdkvis = gdk_x11_screen_lookup_visual( gtk_widget_get_screen(drawable), visual->visualid ); |
nkeynes@659 | 263 | GdkColormap *colormap = gdk_colormap_new( gdkvis, FALSE ); |
nkeynes@659 | 264 | gtk_widget_set_colormap( drawable, colormap ); |
nkeynes@659 | 265 | } |
nkeynes@659 | 266 | #endif |
nkeynes@659 | 267 | gtk_video_drawable = drawable; |
nkeynes@659 | 268 | return drawable; |
nkeynes@659 | 269 | } |
nkeynes@659 | 270 | |
nkeynes@352 | 271 | gboolean video_gtk_init() |
nkeynes@94 | 272 | { |
nkeynes@545 | 273 | |
nkeynes@659 | 274 | if( gtk_video_drawable == NULL ) { |
nkeynes@530 | 275 | return FALSE; |
nkeynes@530 | 276 | } |
nkeynes@435 | 277 | |
nkeynes@659 | 278 | video_width = gtk_video_drawable->allocation.width; |
nkeynes@659 | 279 | video_height = gtk_video_drawable->allocation.height; |
nkeynes@658 | 280 | #ifdef HAVE_OSMESA |
nkeynes@658 | 281 | video_gdk_init_driver( &display_gtk_driver ); |
nkeynes@658 | 282 | #else |
nkeynes@658 | 283 | #ifdef HAVE_GLX |
nkeynes@659 | 284 | Display *display = gdk_x11_display_get_xdisplay( gtk_widget_get_display(GTK_WIDGET(gtk_video_drawable))); |
nkeynes@659 | 285 | Window window = GDK_WINDOW_XWINDOW( GTK_WIDGET(gtk_video_drawable)->window ); |
nkeynes@552 | 286 | if( ! video_glx_init_context( display, window ) || |
nkeynes@552 | 287 | ! video_glx_init_driver( &display_gtk_driver ) ) { |
nkeynes@552 | 288 | return FALSE; |
nkeynes@552 | 289 | } |
nkeynes@663 | 290 | #else |
nkeynes@663 | 291 | #ifdef HAVE_NSGL |
nkeynes@663 | 292 | NSView *view = gdk_quartz_window_get_nsview(gtk_video_drawable->window); |
nkeynes@663 | 293 | if( ! video_nsgl_init_driver( view, &display_gtk_driver ) ) { |
nkeynes@663 | 294 | return FALSE; |
nkeynes@663 | 295 | } |
nkeynes@663 | 296 | #endif |
nkeynes@653 | 297 | #endif |
nkeynes@658 | 298 | #endif |
nkeynes@653 | 299 | |
nkeynes@665 | 300 | pvr2_setup_gl_context(); |
nkeynes@665 | 301 | |
nkeynes@614 | 302 | #ifdef HAVE_LINUX_JOYSTICK |
nkeynes@614 | 303 | linux_joystick_init(); |
nkeynes@614 | 304 | #endif |
nkeynes@552 | 305 | return TRUE; |
nkeynes@552 | 306 | } |
nkeynes@552 | 307 | |
nkeynes@669 | 308 | void video_gtk_display_blank( uint32_t colour ) |
nkeynes@552 | 309 | { |
nkeynes@659 | 310 | GdkGC *gc = gdk_gc_new(gtk_video_drawable->window); |
nkeynes@552 | 311 | GdkColor color = {0, ((colour>>16)&0xFF)*257, ((colour>>8)&0xFF)*257, ((colour)&0xFF)*257 }; |
nkeynes@552 | 312 | GdkColormap *cmap = gdk_colormap_get_system(); |
nkeynes@552 | 313 | gdk_colormap_alloc_color( cmap, &color, TRUE, TRUE ); |
nkeynes@552 | 314 | gdk_gc_set_foreground( gc, &color ); |
nkeynes@552 | 315 | gdk_gc_set_background( gc, &color ); |
nkeynes@659 | 316 | gdk_draw_rectangle( gtk_video_drawable->window, gc, TRUE, 0, 0, video_width, video_height ); |
nkeynes@552 | 317 | gdk_gc_destroy(gc); |
nkeynes@552 | 318 | gdk_colormap_free_colors( cmap, &color, 1 ); |
nkeynes@94 | 319 | } |
nkeynes@94 | 320 | |
nkeynes@352 | 321 | void video_gtk_shutdown() |
nkeynes@352 | 322 | { |
nkeynes@659 | 323 | if( gtk_video_drawable != NULL ) { |
nkeynes@658 | 324 | #ifdef HAVE_OSMESA |
nkeynes@653 | 325 | video_gdk_shutdown(); |
nkeynes@653 | 326 | #else |
nkeynes@659 | 327 | #ifdef HAVE_GLX |
nkeynes@530 | 328 | video_glx_shutdown(); |
nkeynes@663 | 329 | #else |
nkeynes@663 | 330 | #ifdef HAVE_NSGL |
nkeynes@663 | 331 | video_nsgl_shutdown(); |
nkeynes@663 | 332 | #endif |
nkeynes@653 | 333 | #endif |
nkeynes@659 | 334 | #endif |
nkeynes@530 | 335 | } |
nkeynes@620 | 336 | #ifdef HAVE_LINUX_JOYSTICK |
nkeynes@620 | 337 | linux_joystick_shutdown(); |
nkeynes@620 | 338 | #endif |
nkeynes@94 | 339 | } |
nkeynes@94 | 340 |
.