2014-02-10 02:10:30 +01:00
|
|
|
/*************************************************************************/
|
|
|
|
/* os_windows.h */
|
|
|
|
/*************************************************************************/
|
|
|
|
/* This file is part of: */
|
|
|
|
/* GODOT ENGINE */
|
2017-08-27 14:16:55 +02:00
|
|
|
/* https://godotengine.org */
|
2014-02-10 02:10:30 +01:00
|
|
|
/*************************************************************************/
|
2017-01-01 22:01:57 +01:00
|
|
|
/* Copyright (c) 2007-2017 Juan Linietsky, Ariel Manzur. */
|
2017-04-08 00:11:42 +02:00
|
|
|
/* Copyright (c) 2014-2017 Godot Engine contributors (cf. AUTHORS.md) */
|
2014-02-10 02:10:30 +01:00
|
|
|
/* */
|
|
|
|
/* Permission is hereby granted, free of charge, to any person obtaining */
|
|
|
|
/* a copy of this software and associated documentation files (the */
|
|
|
|
/* "Software"), to deal in the Software without restriction, including */
|
|
|
|
/* without limitation the rights to use, copy, modify, merge, publish, */
|
|
|
|
/* distribute, sublicense, and/or sell copies of the Software, and to */
|
|
|
|
/* permit persons to whom the Software is furnished to do so, subject to */
|
|
|
|
/* the following conditions: */
|
|
|
|
/* */
|
|
|
|
/* The above copyright notice and this permission notice shall be */
|
|
|
|
/* included in all copies or substantial portions of the Software. */
|
|
|
|
/* */
|
|
|
|
/* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, */
|
|
|
|
/* EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF */
|
|
|
|
/* MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT.*/
|
|
|
|
/* IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY */
|
|
|
|
/* CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, */
|
|
|
|
/* TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE */
|
|
|
|
/* SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. */
|
|
|
|
/*************************************************************************/
|
|
|
|
#ifndef OS_WINDOWS_H
|
|
|
|
#define OS_WINDOWS_H
|
2017-03-05 16:44:50 +01:00
|
|
|
#include "context_gl_win.h"
|
2017-10-21 13:02:06 +02:00
|
|
|
#include "core/project_settings.h"
|
2017-09-08 03:01:49 +02:00
|
|
|
#include "crash_handler_win.h"
|
2017-03-05 16:44:50 +01:00
|
|
|
#include "drivers/rtaudio/audio_driver_rtaudio.h"
|
2017-08-27 19:01:34 +02:00
|
|
|
#include "drivers/wasapi/audio_driver_wasapi.h"
|
2014-02-10 02:10:30 +01:00
|
|
|
#include "os/input.h"
|
|
|
|
#include "os/os.h"
|
2016-07-23 13:15:55 +02:00
|
|
|
#include "power_windows.h"
|
2017-01-16 19:19:45 +01:00
|
|
|
#include "servers/audio_server.h"
|
2017-03-05 16:44:50 +01:00
|
|
|
#include "servers/visual/rasterizer.h"
|
|
|
|
#include "servers/visual_server.h"
|
2016-10-17 17:40:45 +02:00
|
|
|
#ifdef XAUDIO2_ENABLED
|
|
|
|
#include "drivers/xaudio2/audio_driver_xaudio2.h"
|
|
|
|
#endif
|
2014-02-10 02:10:30 +01:00
|
|
|
#include "drivers/unix/ip_unix.h"
|
2017-03-05 16:44:50 +01:00
|
|
|
#include "key_mapping_win.h"
|
|
|
|
#include "main/input_default.h"
|
2014-02-10 02:10:30 +01:00
|
|
|
|
|
|
|
#include <fcntl.h>
|
2017-03-05 16:44:50 +01:00
|
|
|
#include <io.h>
|
2014-02-10 02:10:30 +01:00
|
|
|
#include <stdio.h>
|
2017-03-05 16:44:50 +01:00
|
|
|
#include <windows.h>
|
|
|
|
#include <windowsx.h>
|
2017-01-16 19:19:45 +01:00
|
|
|
|
2014-02-10 02:10:30 +01:00
|
|
|
/**
|
|
|
|
@author Juan Linietsky <reduzio@gmail.com>
|
|
|
|
*/
|
2017-01-08 23:52:38 +01:00
|
|
|
class JoypadWindows;
|
2014-02-10 02:10:30 +01:00
|
|
|
class OS_Windows : public OS {
|
|
|
|
|
2017-03-05 16:44:50 +01:00
|
|
|
enum {
|
|
|
|
KEY_EVENT_BUFFER_SIZE = 512
|
2014-02-10 02:10:30 +01:00
|
|
|
};
|
|
|
|
|
|
|
|
FILE *stdo;
|
|
|
|
|
|
|
|
struct KeyEvent {
|
|
|
|
|
2017-05-23 13:50:06 +02:00
|
|
|
bool alt, shift, control, meta;
|
2014-02-10 02:10:30 +01:00
|
|
|
UINT uMsg;
|
2017-03-05 16:44:50 +01:00
|
|
|
WPARAM wParam;
|
|
|
|
LPARAM lParam;
|
2014-02-10 02:10:30 +01:00
|
|
|
};
|
|
|
|
|
|
|
|
KeyEvent key_event_buffer[KEY_EVENT_BUFFER_SIZE];
|
|
|
|
int key_event_pos;
|
|
|
|
|
|
|
|
uint64_t ticks_start;
|
|
|
|
uint64_t ticks_per_second;
|
|
|
|
|
2017-03-05 16:44:50 +01:00
|
|
|
bool old_invalid;
|
|
|
|
bool outside;
|
|
|
|
int old_x, old_y;
|
2014-02-10 02:10:30 +01:00
|
|
|
Point2i center;
|
2016-12-21 06:29:58 +01:00
|
|
|
#if defined(OPENGL_ENABLED)
|
2014-02-10 02:10:30 +01:00
|
|
|
ContextGL_Win *gl_context;
|
|
|
|
#endif
|
|
|
|
VisualServer *visual_server;
|
|
|
|
int pressrc;
|
2017-03-05 16:44:50 +01:00
|
|
|
HDC hDC; // Private GDI Device Context
|
|
|
|
HINSTANCE hInstance; // Holds The Instance Of The Application
|
2014-02-10 02:10:30 +01:00
|
|
|
HWND hWnd;
|
2016-05-21 15:29:25 +02:00
|
|
|
|
2016-07-01 15:31:37 +02:00
|
|
|
uint32_t move_timer_id;
|
|
|
|
|
2016-05-09 15:18:08 +02:00
|
|
|
HCURSOR hCursor;
|
2014-02-10 02:10:30 +01:00
|
|
|
|
2015-03-24 00:47:53 +01:00
|
|
|
Size2 window_rect;
|
2014-02-10 02:10:30 +01:00
|
|
|
VideoMode video_mode;
|
|
|
|
|
|
|
|
MainLoop *main_loop;
|
|
|
|
|
|
|
|
WNDPROC user_proc;
|
|
|
|
|
|
|
|
MouseMode mouse_mode;
|
|
|
|
bool alt_mem;
|
|
|
|
bool gr_mem;
|
|
|
|
bool shift_mem;
|
|
|
|
bool control_mem;
|
|
|
|
bool meta_mem;
|
|
|
|
bool force_quit;
|
2017-01-25 19:21:41 +01:00
|
|
|
bool window_has_focus;
|
2014-02-10 02:10:30 +01:00
|
|
|
uint32_t last_button_state;
|
|
|
|
|
|
|
|
CursorShape cursor_shape;
|
|
|
|
|
|
|
|
InputDefault *input;
|
2017-01-08 23:52:38 +01:00
|
|
|
JoypadWindows *joypad;
|
2017-11-30 19:55:38 +01:00
|
|
|
Map<int, Vector2> touch_state;
|
2014-02-10 02:10:30 +01:00
|
|
|
|
2016-07-23 13:15:55 +02:00
|
|
|
PowerWindows *power_manager;
|
|
|
|
|
2017-08-27 19:01:34 +02:00
|
|
|
#ifdef WASAPI_ENABLED
|
|
|
|
AudioDriverWASAPI driver_wasapi;
|
|
|
|
#endif
|
2014-02-10 02:10:30 +01:00
|
|
|
#ifdef RTAUDIO_ENABLED
|
|
|
|
AudioDriverRtAudio driver_rtaudio;
|
|
|
|
#endif
|
2016-10-17 17:40:45 +02:00
|
|
|
#ifdef XAUDIO2_ENABLED
|
|
|
|
AudioDriverXAudio2 driver_xaudio2;
|
|
|
|
#endif
|
2014-02-10 02:10:30 +01:00
|
|
|
|
2017-09-08 03:01:49 +02:00
|
|
|
CrashHandler crash_handler;
|
|
|
|
|
2017-11-30 19:55:38 +01:00
|
|
|
void _drag_event(float p_x, float p_y, int idx);
|
|
|
|
void _touch_event(bool p_pressed, float p_x, float p_y, int idx);
|
2014-09-03 04:13:40 +02:00
|
|
|
|
2017-07-10 02:48:22 +02:00
|
|
|
void _update_window_style(bool repaint = true);
|
|
|
|
|
2014-02-10 02:10:30 +01:00
|
|
|
// functions used by main to initialize/deintialize the OS
|
2016-03-09 00:00:52 +01:00
|
|
|
protected:
|
2014-02-10 02:10:30 +01:00
|
|
|
virtual int get_video_driver_count() const;
|
2017-03-05 16:44:50 +01:00
|
|
|
virtual const char *get_video_driver_name(int p_driver) const;
|
2016-03-09 00:00:52 +01:00
|
|
|
|
2014-02-10 02:10:30 +01:00
|
|
|
virtual int get_audio_driver_count() const;
|
2017-03-05 16:44:50 +01:00
|
|
|
virtual const char *get_audio_driver_name(int p_driver) const;
|
2016-03-09 00:00:52 +01:00
|
|
|
|
2014-02-10 02:10:30 +01:00
|
|
|
virtual void initialize_core();
|
2017-03-05 16:44:50 +01:00
|
|
|
virtual void initialize(const VideoMode &p_desired, int p_video_driver, int p_audio_driver);
|
2016-03-09 00:00:52 +01:00
|
|
|
|
2017-03-05 16:44:50 +01:00
|
|
|
virtual void set_main_loop(MainLoop *p_main_loop);
|
2014-02-10 02:10:30 +01:00
|
|
|
virtual void delete_main_loop();
|
2016-03-09 00:00:52 +01:00
|
|
|
|
2014-02-10 02:10:30 +01:00
|
|
|
virtual void finalize();
|
|
|
|
virtual void finalize_core();
|
2016-03-09 00:00:52 +01:00
|
|
|
|
2014-02-10 02:10:30 +01:00
|
|
|
void process_events();
|
|
|
|
void process_key_events();
|
2016-03-09 00:00:52 +01:00
|
|
|
|
2014-02-10 02:10:30 +01:00
|
|
|
struct ProcessInfo {
|
|
|
|
|
|
|
|
STARTUPINFO si;
|
|
|
|
PROCESS_INFORMATION pi;
|
|
|
|
};
|
2017-03-05 16:44:50 +01:00
|
|
|
Map<ProcessID, ProcessInfo> *process_map;
|
2014-02-10 02:10:30 +01:00
|
|
|
|
2015-03-24 00:47:53 +01:00
|
|
|
bool pre_fs_valid;
|
2015-03-22 23:00:50 +01:00
|
|
|
RECT pre_fs_rect;
|
|
|
|
bool maximized;
|
|
|
|
bool minimized;
|
2016-01-03 05:18:28 +01:00
|
|
|
bool borderless;
|
2015-03-22 23:00:50 +01:00
|
|
|
|
2014-02-10 02:10:30 +01:00
|
|
|
public:
|
2017-03-05 16:44:50 +01:00
|
|
|
LRESULT WndProc(HWND hWnd, UINT uMsg, WPARAM wParam, LPARAM lParam);
|
2014-02-10 02:10:30 +01:00
|
|
|
|
2017-03-05 16:44:50 +01:00
|
|
|
virtual void alert(const String &p_alert, const String &p_title = "ALERT!");
|
2014-02-10 02:10:30 +01:00
|
|
|
String get_stdin_string(bool p_block);
|
|
|
|
|
|
|
|
void set_mouse_mode(MouseMode p_mode);
|
|
|
|
MouseMode get_mouse_mode() const;
|
|
|
|
|
2017-09-10 15:37:49 +02:00
|
|
|
virtual void warp_mouse_position(const Point2 &p_to);
|
2017-03-29 17:29:38 +02:00
|
|
|
virtual Point2 get_mouse_position() const;
|
2014-02-10 02:10:30 +01:00
|
|
|
virtual int get_mouse_button_state() const;
|
2017-03-05 16:44:50 +01:00
|
|
|
virtual void set_window_title(const String &p_title);
|
2016-03-09 00:00:52 +01:00
|
|
|
|
2017-03-05 16:44:50 +01:00
|
|
|
virtual void set_video_mode(const VideoMode &p_video_mode, int p_screen = 0);
|
|
|
|
virtual VideoMode get_video_mode(int p_screen = 0) const;
|
|
|
|
virtual void get_fullscreen_mode_list(List<VideoMode> *p_list, int p_screen = 0) const;
|
2014-02-10 02:10:30 +01:00
|
|
|
|
2015-03-22 23:00:50 +01:00
|
|
|
virtual int get_screen_count() const;
|
|
|
|
virtual int get_current_screen() const;
|
|
|
|
virtual void set_current_screen(int p_screen);
|
2017-08-20 21:12:29 +02:00
|
|
|
virtual Point2 get_screen_position(int p_screen = -1) const;
|
|
|
|
virtual Size2 get_screen_size(int p_screen = -1) const;
|
|
|
|
virtual int get_screen_dpi(int p_screen = -1) const;
|
2016-05-29 18:40:00 +02:00
|
|
|
|
2015-03-22 23:00:50 +01:00
|
|
|
virtual Point2 get_window_position() const;
|
2017-03-05 16:44:50 +01:00
|
|
|
virtual void set_window_position(const Point2 &p_position);
|
2015-03-22 23:00:50 +01:00
|
|
|
virtual Size2 get_window_size() const;
|
|
|
|
virtual void set_window_size(const Size2 p_size);
|
|
|
|
virtual void set_window_fullscreen(bool p_enabled);
|
|
|
|
virtual bool is_window_fullscreen() const;
|
|
|
|
virtual void set_window_resizable(bool p_enabled);
|
|
|
|
virtual bool is_window_resizable() const;
|
|
|
|
virtual void set_window_minimized(bool p_enabled);
|
|
|
|
virtual bool is_window_minimized() const;
|
|
|
|
virtual void set_window_maximized(bool p_enabled);
|
|
|
|
virtual bool is_window_maximized() const;
|
2016-07-05 17:29:08 +02:00
|
|
|
virtual void request_attention();
|
2015-03-22 23:00:50 +01:00
|
|
|
|
2017-12-14 19:15:46 +01:00
|
|
|
virtual void set_borderless_window(bool p_borderless);
|
2016-01-03 05:18:28 +01:00
|
|
|
virtual bool get_borderless_window();
|
|
|
|
|
2017-12-01 21:06:42 +01:00
|
|
|
virtual Error open_dynamic_library(const String p_path, void *&p_library_handle, bool p_also_set_library_path = false);
|
2017-04-03 16:11:38 +02:00
|
|
|
virtual Error close_dynamic_library(void *p_library_handle);
|
2017-07-27 09:23:21 +02:00
|
|
|
virtual Error get_dynamic_library_symbol_handle(void *p_library_handle, const String p_name, void *&p_symbol_handle, bool p_optional = false);
|
2017-03-08 02:50:13 +01:00
|
|
|
|
2014-02-10 02:10:30 +01:00
|
|
|
virtual MainLoop *get_main_loop() const;
|
|
|
|
|
|
|
|
virtual String get_name();
|
2016-03-09 00:00:52 +01:00
|
|
|
|
2015-06-06 03:40:56 +02:00
|
|
|
virtual Date get_date(bool utc) const;
|
|
|
|
virtual Time get_time(bool utc) const;
|
2015-06-07 15:06:13 +02:00
|
|
|
virtual TimeZoneInfo get_time_zone_info() const;
|
2014-02-10 02:10:30 +01:00
|
|
|
virtual uint64_t get_unix_time() const;
|
2016-01-10 22:24:55 +01:00
|
|
|
virtual uint64_t get_system_time_secs() const;
|
2014-02-10 02:10:30 +01:00
|
|
|
|
|
|
|
virtual bool can_draw() const;
|
2017-03-05 16:44:50 +01:00
|
|
|
virtual Error set_cwd(const String &p_cwd);
|
2014-02-10 02:10:30 +01:00
|
|
|
|
2016-03-09 00:00:52 +01:00
|
|
|
virtual void delay_usec(uint32_t p_usec) const;
|
2014-02-10 02:10:30 +01:00
|
|
|
virtual uint64_t get_ticks_usec() const;
|
|
|
|
|
2017-10-03 20:09:04 +02:00
|
|
|
virtual Error execute(const String &p_path, const List<String> &p_arguments, bool p_blocking, ProcessID *r_child_id = NULL, String *r_pipe = NULL, int *r_exitcode = NULL, bool read_stderr = false);
|
2017-03-05 16:44:50 +01:00
|
|
|
virtual Error kill(const ProcessID &p_pid);
|
2017-08-07 12:17:31 +02:00
|
|
|
virtual int get_process_id() const;
|
2016-03-09 00:00:52 +01:00
|
|
|
|
2017-03-05 16:44:50 +01:00
|
|
|
virtual bool has_environment(const String &p_var) const;
|
|
|
|
virtual String get_environment(const String &p_var) const;
|
2014-02-10 02:10:30 +01:00
|
|
|
|
2017-03-05 16:44:50 +01:00
|
|
|
virtual void set_clipboard(const String &p_text);
|
2014-02-10 02:10:30 +01:00
|
|
|
virtual String get_clipboard() const;
|
|
|
|
|
|
|
|
void set_cursor_shape(CursorShape p_shape);
|
2017-05-17 12:36:47 +02:00
|
|
|
void set_icon(const Ref<Image> &p_icon);
|
2014-02-10 02:10:30 +01:00
|
|
|
|
2014-02-15 06:01:39 +01:00
|
|
|
virtual String get_executable_path() const;
|
2014-02-10 02:10:30 +01:00
|
|
|
|
|
|
|
virtual String get_locale() const;
|
2017-03-05 16:44:50 +01:00
|
|
|
virtual LatinKeyboardVariant get_latin_keyboard_variant() const;
|
2014-02-10 02:10:30 +01:00
|
|
|
|
2016-09-14 04:02:18 +02:00
|
|
|
virtual void enable_for_stealing_focus(ProcessID pid);
|
2014-02-10 02:10:30 +01:00
|
|
|
virtual void move_window_to_foreground();
|
Add initial support for the XDG Base Directory spec
Spec version 0.7 from https://standards.freedesktop.org/basedir-spec/basedir-spec-0.7.html
(latest as of this commit).
Three virtual methods are added to OS for the various XDG paths we will use:
- OS::get_data_path gives XDG_DATA_HOME, or if missing:
~/.local/share on X11, ~/Library/Application Support/ on macOS and %APPDATA% on Windows
- OS::get_config_path gives XDG_CONFIG_HOME, or if missing:
~/.config on X11, ~/Library/Application Support/ on macOS and %APPDATA% on Windows
- OS::get_cache_path gives XDG_CACHE_HOME, or if missing:
~/.cache on X11, ~/Library/Caches on macOS and %APPDATA% on Windows
So for Windows there are no changes, for Linux we follow the full split spec
and for macOS stuff will move from ~/.godot to ~/Library/Application Support/Godot.
Support for system-wide installation of templates on Unix was removed for now,
as it's a bit hackish and I don't think anyone uses it.
user:// will still be OS::get_data_path() + "/godot/app_userdata/$name" by
default, but when using the application/config/use_shared_user_dir option
it will now use XDG_DATA_HOME/$name, e.g. ~/.local/share/MyGame.
For now everything still goes in EditorSettings::get_settings_dir(), but
this will be changed in a later commit to make use of the new splitting
where relevant.
Part of #3513.
2017-11-17 17:11:41 +01:00
|
|
|
|
|
|
|
virtual String get_config_path() const;
|
|
|
|
virtual String get_data_path() const;
|
|
|
|
virtual String get_cache_path() const;
|
|
|
|
virtual String get_godot_dir_name() const;
|
|
|
|
|
2014-12-02 18:02:41 +01:00
|
|
|
virtual String get_system_dir(SystemDir p_dir) const;
|
Add initial support for the XDG Base Directory spec
Spec version 0.7 from https://standards.freedesktop.org/basedir-spec/basedir-spec-0.7.html
(latest as of this commit).
Three virtual methods are added to OS for the various XDG paths we will use:
- OS::get_data_path gives XDG_DATA_HOME, or if missing:
~/.local/share on X11, ~/Library/Application Support/ on macOS and %APPDATA% on Windows
- OS::get_config_path gives XDG_CONFIG_HOME, or if missing:
~/.config on X11, ~/Library/Application Support/ on macOS and %APPDATA% on Windows
- OS::get_cache_path gives XDG_CACHE_HOME, or if missing:
~/.cache on X11, ~/Library/Caches on macOS and %APPDATA% on Windows
So for Windows there are no changes, for Linux we follow the full split spec
and for macOS stuff will move from ~/.godot to ~/Library/Application Support/Godot.
Support for system-wide installation of templates on Unix was removed for now,
as it's a bit hackish and I don't think anyone uses it.
user:// will still be OS::get_data_path() + "/godot/app_userdata/$name" by
default, but when using the application/config/use_shared_user_dir option
it will now use XDG_DATA_HOME/$name, e.g. ~/.local/share/MyGame.
For now everything still goes in EditorSettings::get_settings_dir(), but
this will be changed in a later commit to make use of the new splitting
where relevant.
Part of #3513.
2017-11-17 17:11:41 +01:00
|
|
|
virtual String get_user_data_dir() const;
|
2014-12-02 18:02:41 +01:00
|
|
|
|
2014-02-10 02:10:30 +01:00
|
|
|
virtual void release_rendering_thread();
|
|
|
|
virtual void make_rendering_thread();
|
|
|
|
virtual void swap_buffers();
|
|
|
|
|
2014-04-18 16:43:54 +02:00
|
|
|
virtual Error shell_open(String p_uri);
|
|
|
|
|
2014-02-10 02:10:30 +01:00
|
|
|
void run();
|
|
|
|
|
|
|
|
virtual bool get_swap_ok_cancel() { return true; }
|
|
|
|
|
2016-01-08 00:40:41 +01:00
|
|
|
virtual bool is_joy_known(int p_device);
|
|
|
|
virtual String get_joy_guid(int p_device) const;
|
|
|
|
|
2016-06-06 00:14:33 +02:00
|
|
|
virtual void set_use_vsync(bool p_enable);
|
2016-11-08 15:06:57 +01:00
|
|
|
virtual bool is_vsync_enabled() const;
|
2016-06-06 00:14:33 +02:00
|
|
|
|
2017-09-12 21:09:06 +02:00
|
|
|
virtual OS::PowerState get_power_state();
|
2016-07-23 13:15:55 +02:00
|
|
|
virtual int get_power_seconds_left();
|
|
|
|
virtual int get_power_percent_left();
|
|
|
|
|
2017-07-19 22:00:46 +02:00
|
|
|
virtual bool _check_internal_feature_support(const String &p_feature);
|
2017-02-09 00:07:35 +01:00
|
|
|
|
2017-09-08 03:01:49 +02:00
|
|
|
void disable_crash_handler();
|
|
|
|
bool is_disable_crash_handler() const;
|
|
|
|
|
2017-12-14 12:59:46 +01:00
|
|
|
void force_process_input();
|
|
|
|
|
2017-09-25 15:15:11 +02:00
|
|
|
virtual Error move_to_trash(const String &p_path);
|
|
|
|
|
2016-03-09 00:00:52 +01:00
|
|
|
OS_Windows(HINSTANCE _hInstance);
|
2014-02-10 02:10:30 +01:00
|
|
|
~OS_Windows();
|
|
|
|
};
|
|
|
|
|
|
|
|
#endif
|