2018-08-31 15:50:20 +01:00
|
|
|
/*
|
|
|
|
* Copyright 2018 Collabora Ltd.
|
|
|
|
*
|
|
|
|
* 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
|
|
|
|
* on the rights to use, copy, modify, merge, publish, distribute, sub
|
|
|
|
* license, 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 (including the next
|
|
|
|
* paragraph) 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 NON-INFRINGEMENT. IN NO EVENT SHALL
|
|
|
|
* THE AUTHOR(S) AND/OR THEIR SUPPLIERS 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 ZINK_SCREEN_H
|
|
|
|
#define ZINK_SCREEN_H
|
|
|
|
|
2020-09-26 07:49:49 +01:00
|
|
|
#include "zink_device_info.h"
|
2020-11-26 17:32:39 +00:00
|
|
|
#include "zink_instance.h"
|
2021-06-13 18:36:53 +01:00
|
|
|
#include "vk_dispatch_table.h"
|
2020-09-26 07:49:49 +01:00
|
|
|
|
2021-05-19 15:08:22 +01:00
|
|
|
#include "util/u_idalloc.h"
|
2018-08-31 15:50:20 +01:00
|
|
|
#include "pipe/p_screen.h"
|
|
|
|
#include "util/slab.h"
|
2021-01-06 08:57:49 +00:00
|
|
|
#include "compiler/nir/nir.h"
|
2020-10-06 16:42:27 +01:00
|
|
|
#include "util/disk_cache.h"
|
2021-02-19 06:07:56 +00:00
|
|
|
#include "util/log.h"
|
2020-10-13 14:42:07 +01:00
|
|
|
#include "util/simple_mtx.h"
|
2021-05-09 17:55:39 +01:00
|
|
|
#include "util/u_queue.h"
|
2021-05-10 11:50:49 +01:00
|
|
|
#include "util/u_live_shader_cache.h"
|
2021-11-05 16:00:25 +00:00
|
|
|
#include "util/u_vertex_state_cache.h"
|
2021-05-13 12:33:44 +01:00
|
|
|
#include "pipebuffer/pb_cache.h"
|
|
|
|
#include "pipebuffer/pb_slab.h"
|
2022-01-13 20:29:54 +00:00
|
|
|
|
2018-08-31 15:50:20 +01:00
|
|
|
#include <vulkan/vulkan.h>
|
|
|
|
|
2022-05-16 16:27:58 +01:00
|
|
|
|
|
|
|
#ifdef __cplusplus
|
|
|
|
extern "C" {
|
|
|
|
#endif
|
|
|
|
|
2018-08-31 15:50:20 +01:00
|
|
|
extern uint32_t zink_debug;
|
2020-10-13 14:42:07 +01:00
|
|
|
struct hash_table;
|
2021-06-23 09:23:38 +01:00
|
|
|
struct util_dl_library;
|
2018-08-31 15:50:20 +01:00
|
|
|
|
2021-04-01 20:18:51 +01:00
|
|
|
struct zink_batch_state;
|
|
|
|
struct zink_context;
|
|
|
|
struct zink_descriptor_layout_key;
|
|
|
|
struct zink_program;
|
|
|
|
struct zink_shader;
|
|
|
|
enum zink_descriptor_type;
|
|
|
|
|
2021-05-13 12:33:44 +01:00
|
|
|
/* this is the spec minimum */
|
|
|
|
#define ZINK_SPARSE_BUFFER_PAGE_SIZE (64 * 1024)
|
|
|
|
|
2022-07-15 16:05:50 +01:00
|
|
|
enum zink_debug {
|
|
|
|
ZINK_DEBUG_NIR = (1<<0),
|
|
|
|
ZINK_DEBUG_SPIRV = (1<<1),
|
|
|
|
ZINK_DEBUG_TGSI = (1<<2),
|
|
|
|
ZINK_DEBUG_VALIDATION = (1<<3),
|
|
|
|
ZINK_DEBUG_SYNC = (1<<4),
|
|
|
|
ZINK_DEBUG_COMPACT = (1<<5),
|
2022-07-15 16:09:04 +01:00
|
|
|
ZINK_DEBUG_NOREORDER = (1<<6),
|
2022-07-15 16:05:50 +01:00
|
|
|
};
|
2018-08-31 15:50:20 +01:00
|
|
|
|
2021-05-13 12:33:44 +01:00
|
|
|
#define NUM_SLAB_ALLOCATORS 3
|
2022-01-25 18:47:12 +00:00
|
|
|
#define MIN_SLAB_ORDER 8
|
2021-05-13 12:33:44 +01:00
|
|
|
|
2022-04-29 12:41:32 +01:00
|
|
|
#define ZINK_CONTEXT_COPY_ONLY (1<<30)
|
|
|
|
|
2021-03-29 22:22:49 +01:00
|
|
|
enum zink_descriptor_mode {
|
|
|
|
ZINK_DESCRIPTOR_MODE_AUTO,
|
|
|
|
ZINK_DESCRIPTOR_MODE_LAZY,
|
2022-07-19 15:08:07 +01:00
|
|
|
ZINK_DESCRIPTOR_MODE_CACHED,
|
2021-03-29 22:22:49 +01:00
|
|
|
ZINK_DESCRIPTOR_MODE_NOTEMPLATES,
|
2022-05-16 22:28:59 +01:00
|
|
|
ZINK_DESCRIPTOR_MODE_COMPACT,
|
2021-03-29 22:22:49 +01:00
|
|
|
};
|
|
|
|
|
2022-07-19 14:58:03 +01:00
|
|
|
extern enum zink_descriptor_mode zink_descriptor_mode;
|
|
|
|
|
2022-05-16 22:28:59 +01:00
|
|
|
//keep in sync with zink_descriptor_type since headers can't be cross-included
|
|
|
|
#define ZINK_MAX_DESCRIPTOR_SETS 6
|
|
|
|
|
2021-04-08 21:10:24 +01:00
|
|
|
struct zink_modifier_prop {
|
|
|
|
uint32_t drmFormatModifierCount;
|
|
|
|
VkDrmFormatModifierPropertiesEXT* pDrmFormatModifierProperties;
|
|
|
|
};
|
|
|
|
|
2018-08-31 15:50:20 +01:00
|
|
|
struct zink_screen {
|
|
|
|
struct pipe_screen base;
|
2021-06-23 09:23:38 +01:00
|
|
|
|
|
|
|
struct util_dl_library *loader_lib;
|
|
|
|
PFN_vkGetInstanceProcAddr vk_GetInstanceProcAddr;
|
|
|
|
PFN_vkGetDeviceProcAddr vk_GetDeviceProcAddr;
|
|
|
|
|
2021-03-22 15:01:30 +00:00
|
|
|
bool threaded;
|
2021-11-09 15:44:45 +00:00
|
|
|
bool is_cpu;
|
2022-07-12 14:17:25 +01:00
|
|
|
bool abort_on_hang;
|
2022-03-25 18:00:21 +00:00
|
|
|
uint64_t curr_batch; //the current batch id
|
|
|
|
uint32_t last_finished;
|
2021-04-08 15:52:55 +01:00
|
|
|
VkSemaphore sem;
|
2022-06-10 18:24:20 +01:00
|
|
|
VkFence fence;
|
2021-05-09 17:55:39 +01:00
|
|
|
struct util_queue flush_queue;
|
2022-04-29 12:41:52 +01:00
|
|
|
struct zink_context *copy_context;
|
2018-08-31 15:50:20 +01:00
|
|
|
|
2021-06-03 20:20:34 +01:00
|
|
|
unsigned buffer_rebind_counter;
|
2022-05-09 17:23:26 +01:00
|
|
|
unsigned image_rebind_counter;
|
2022-07-12 14:17:25 +01:00
|
|
|
unsigned robust_ctx_count;
|
2021-06-03 20:20:34 +01:00
|
|
|
|
2022-01-13 20:29:54 +00:00
|
|
|
struct hash_table dts;
|
|
|
|
simple_mtx_t dt_lock;
|
|
|
|
|
2021-02-10 23:40:55 +00:00
|
|
|
bool device_lost;
|
2022-06-01 16:30:32 +01:00
|
|
|
int drm_fd;
|
2018-08-31 15:50:20 +01:00
|
|
|
|
2021-02-09 21:22:33 +00:00
|
|
|
struct hash_table framebuffer_cache;
|
2021-02-09 20:40:38 +00:00
|
|
|
|
2018-08-31 15:50:20 +01:00
|
|
|
struct slab_parent_pool transfer_pool;
|
2020-10-06 16:42:27 +01:00
|
|
|
struct disk_cache *disk_cache;
|
2021-05-09 16:13:09 +01:00
|
|
|
struct util_queue cache_put_thread;
|
|
|
|
struct util_queue cache_get_thread;
|
2018-08-31 15:50:20 +01:00
|
|
|
|
2021-05-10 11:50:49 +01:00
|
|
|
struct util_live_shader_cache shaders;
|
|
|
|
|
2021-05-13 12:33:44 +01:00
|
|
|
struct {
|
|
|
|
struct pb_cache bo_cache;
|
|
|
|
struct pb_slabs bo_slabs[NUM_SLAB_ALLOCATORS];
|
|
|
|
unsigned min_alloc_size;
|
|
|
|
uint32_t next_bo_unique_id;
|
|
|
|
} pb;
|
|
|
|
uint8_t heap_map[VK_MAX_MEMORY_TYPES];
|
2022-06-16 16:06:53 +01:00
|
|
|
VkMemoryPropertyFlags heap_flags[VK_MAX_MEMORY_TYPES];
|
2021-07-30 14:18:54 +01:00
|
|
|
bool resizable_bar;
|
2020-10-13 14:42:07 +01:00
|
|
|
|
2021-04-02 16:08:46 +01:00
|
|
|
uint64_t total_video_mem;
|
2021-05-15 15:20:43 +01:00
|
|
|
uint64_t clamp_video_mem;
|
2020-09-28 15:50:23 +01:00
|
|
|
uint64_t total_mem;
|
|
|
|
|
2018-08-31 15:50:20 +01:00
|
|
|
VkInstance instance;
|
2020-11-26 17:32:39 +00:00
|
|
|
struct zink_instance_info instance_info;
|
|
|
|
|
2018-08-31 15:50:20 +01:00
|
|
|
VkPhysicalDevice pdev;
|
2021-05-27 13:48:52 +01:00
|
|
|
uint32_t vk_version, spirv_version;
|
2021-05-19 15:08:22 +01:00
|
|
|
struct util_idalloc_mt buffer_ids;
|
2021-11-05 16:00:25 +00:00
|
|
|
struct util_vertex_state_cache vertex_state_cache;
|
2018-10-24 08:46:05 +01:00
|
|
|
|
2020-09-26 07:49:49 +01:00
|
|
|
struct zink_device_info info;
|
2021-01-06 08:57:49 +00:00
|
|
|
struct nir_shader_compiler_options nir_options;
|
2020-09-26 07:49:49 +01:00
|
|
|
|
2019-10-23 10:59:03 +01:00
|
|
|
bool have_X8_D24_UNORM_PACK32;
|
|
|
|
bool have_D24_UNORM_S8_UINT;
|
2020-11-05 11:17:40 +00:00
|
|
|
bool have_triangle_fans;
|
2021-10-28 20:17:11 +01:00
|
|
|
bool need_2D_zs;
|
2022-04-21 17:24:01 +01:00
|
|
|
bool need_2D_sparse;
|
2022-01-03 19:32:32 +00:00
|
|
|
bool faked_e5sparse; //drivers may not expose R9G9B9E5 but cts requires it
|
2019-10-23 10:59:03 +01:00
|
|
|
|
2018-08-31 15:50:20 +01:00
|
|
|
uint32_t gfx_queue;
|
2022-06-10 20:32:58 +01:00
|
|
|
uint32_t sparse_queue;
|
2021-03-22 15:01:30 +00:00
|
|
|
uint32_t max_queues;
|
2020-07-10 19:40:01 +01:00
|
|
|
uint32_t timestamp_valid_bits;
|
2018-08-31 15:50:20 +01:00
|
|
|
VkDevice dev;
|
2021-05-06 21:41:36 +01:00
|
|
|
VkQueue queue; //gfx+compute
|
2022-06-10 20:32:58 +01:00
|
|
|
VkQueue queue_sparse;
|
2021-10-29 16:06:34 +01:00
|
|
|
simple_mtx_t queue_lock;
|
2020-11-04 15:41:11 +00:00
|
|
|
VkDebugUtilsMessengerEXT debugUtilsCallbackHandle;
|
2018-10-02 23:27:36 +01:00
|
|
|
|
2020-08-04 20:42:43 +01:00
|
|
|
uint32_t cur_custom_border_color_samplers;
|
|
|
|
|
2021-05-26 20:10:09 +01:00
|
|
|
struct vk_dispatch_table vk;
|
|
|
|
|
2022-05-16 22:28:59 +01:00
|
|
|
bool compact_descriptors;
|
2022-05-16 21:54:47 +01:00
|
|
|
uint8_t desc_set_id[ZINK_MAX_DESCRIPTOR_SETS];
|
2021-04-01 20:18:51 +01:00
|
|
|
bool (*descriptor_program_init)(struct zink_context *ctx, struct zink_program *pg);
|
2021-10-13 15:03:34 +01:00
|
|
|
void (*descriptor_program_deinit)(struct zink_context *ctx, struct zink_program *pg);
|
2021-05-21 21:54:46 +01:00
|
|
|
void (*descriptors_update)(struct zink_context *ctx, bool is_compute);
|
2021-04-01 20:18:51 +01:00
|
|
|
void (*context_update_descriptor_states)(struct zink_context *ctx, bool is_compute);
|
|
|
|
void (*context_invalidate_descriptor_state)(struct zink_context *ctx, enum pipe_shader_type shader,
|
|
|
|
enum zink_descriptor_type type,
|
|
|
|
unsigned start, unsigned count);
|
|
|
|
bool (*batch_descriptor_init)(struct zink_screen *screen, struct zink_batch_state *bs);
|
|
|
|
void (*batch_descriptor_reset)(struct zink_screen *screen, struct zink_batch_state *bs);
|
|
|
|
void (*batch_descriptor_deinit)(struct zink_screen *screen, struct zink_batch_state *bs);
|
|
|
|
bool (*descriptors_init)(struct zink_context *ctx);
|
|
|
|
void (*descriptors_deinit)(struct zink_context *ctx);
|
2021-01-08 12:59:51 +00:00
|
|
|
|
2020-09-23 20:48:27 +01:00
|
|
|
struct {
|
|
|
|
bool dual_color_blend_by_location;
|
2020-12-16 17:20:39 +00:00
|
|
|
bool inline_uniforms;
|
2020-09-23 20:48:27 +01:00
|
|
|
} driconf;
|
2020-10-07 17:45:04 +01:00
|
|
|
|
|
|
|
VkFormatProperties format_props[PIPE_FORMAT_COUNT];
|
2021-04-08 21:10:24 +01:00
|
|
|
struct zink_modifier_prop modifier_props[PIPE_FORMAT_COUNT];
|
2020-10-07 18:59:35 +01:00
|
|
|
struct {
|
2020-11-05 15:39:19 +00:00
|
|
|
uint32_t image_view;
|
|
|
|
uint32_t buffer_view;
|
2020-10-07 18:59:35 +01:00
|
|
|
} null_descriptor_hashes;
|
2021-04-02 20:15:45 +01:00
|
|
|
|
2021-04-02 20:16:17 +01:00
|
|
|
VkExtent2D maxSampleLocationGridSize[5];
|
2022-03-13 16:11:33 +00:00
|
|
|
|
|
|
|
struct {
|
|
|
|
bool color_write_missing;
|
2022-03-22 22:05:35 +00:00
|
|
|
bool depth_clip_control_missing;
|
2022-06-10 18:24:20 +01:00
|
|
|
bool implicit_sync;
|
2022-03-13 16:11:33 +00:00
|
|
|
} driver_workarounds;
|
2018-08-31 15:50:20 +01:00
|
|
|
};
|
|
|
|
|
2020-12-29 17:10:08 +00:00
|
|
|
/* update last_finished to account for batch_id wrapping */
|
|
|
|
static inline void
|
2022-03-25 18:00:21 +00:00
|
|
|
zink_screen_update_last_finished(struct zink_screen *screen, uint64_t batch_id)
|
2020-12-29 17:10:08 +00:00
|
|
|
{
|
2022-03-25 18:00:21 +00:00
|
|
|
const uint32_t check_id = (uint32_t)batch_id;
|
2020-12-29 17:10:08 +00:00
|
|
|
/* last_finished may have wrapped */
|
|
|
|
if (screen->last_finished < UINT_MAX / 2) {
|
|
|
|
/* last_finished has wrapped, batch_id has not */
|
2022-03-25 18:00:21 +00:00
|
|
|
if (check_id > UINT_MAX / 2)
|
2020-12-29 17:10:08 +00:00
|
|
|
return;
|
2022-03-25 18:00:21 +00:00
|
|
|
} else if (check_id < UINT_MAX / 2) {
|
2020-12-29 17:10:08 +00:00
|
|
|
/* batch_id has wrapped, last_finished has not */
|
2022-03-25 18:00:21 +00:00
|
|
|
screen->last_finished = check_id;
|
2020-12-29 17:10:08 +00:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
/* neither have wrapped */
|
2022-03-25 18:00:21 +00:00
|
|
|
screen->last_finished = MAX2(check_id, screen->last_finished);
|
2020-12-29 17:10:08 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/* check a batch_id against last_finished while accounting for wrapping */
|
|
|
|
static inline bool
|
|
|
|
zink_screen_check_last_finished(struct zink_screen *screen, uint32_t batch_id)
|
|
|
|
{
|
2022-03-25 18:00:21 +00:00
|
|
|
const uint32_t check_id = (uint32_t)batch_id;
|
2020-12-29 17:10:08 +00:00
|
|
|
/* last_finished may have wrapped */
|
|
|
|
if (screen->last_finished < UINT_MAX / 2) {
|
|
|
|
/* last_finished has wrapped, batch_id has not */
|
2022-03-25 18:00:21 +00:00
|
|
|
if (check_id > UINT_MAX / 2)
|
2020-12-29 17:10:08 +00:00
|
|
|
return true;
|
2022-03-25 18:00:21 +00:00
|
|
|
} else if (check_id < UINT_MAX / 2) {
|
2020-12-29 17:10:08 +00:00
|
|
|
/* batch_id has wrapped, last_finished has not */
|
|
|
|
return false;
|
|
|
|
}
|
2022-03-25 18:00:21 +00:00
|
|
|
return screen->last_finished >= check_id;
|
2020-12-29 17:10:08 +00:00
|
|
|
}
|
|
|
|
|
2021-04-08 15:52:55 +01:00
|
|
|
bool
|
|
|
|
zink_screen_init_semaphore(struct zink_screen *screen);
|
|
|
|
|
2021-04-04 17:59:41 +01:00
|
|
|
static inline bool
|
|
|
|
zink_screen_handle_vkresult(struct zink_screen *screen, VkResult ret)
|
|
|
|
{
|
|
|
|
bool success = false;
|
|
|
|
switch (ret) {
|
|
|
|
case VK_SUCCESS:
|
|
|
|
success = true;
|
|
|
|
break;
|
|
|
|
case VK_ERROR_DEVICE_LOST:
|
|
|
|
screen->device_lost = true;
|
2022-01-18 22:40:36 +00:00
|
|
|
mesa_loge("zink: DEVICE LOST!\n");
|
2022-07-12 14:17:25 +01:00
|
|
|
/* if nothing can save us, abort */
|
|
|
|
if (screen->abort_on_hang && !screen->robust_ctx_count)
|
|
|
|
abort();
|
2021-04-04 17:59:41 +01:00
|
|
|
FALLTHROUGH;
|
|
|
|
default:
|
|
|
|
success = false;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
return success;
|
|
|
|
}
|
2021-04-08 15:52:55 +01:00
|
|
|
|
2018-08-31 15:50:20 +01:00
|
|
|
static inline struct zink_screen *
|
|
|
|
zink_screen(struct pipe_screen *pipe)
|
|
|
|
{
|
|
|
|
return (struct zink_screen *)pipe;
|
|
|
|
}
|
|
|
|
|
2021-01-26 23:10:13 +00:00
|
|
|
|
|
|
|
struct mem_cache_entry {
|
|
|
|
VkDeviceMemory mem;
|
|
|
|
void *map;
|
|
|
|
};
|
|
|
|
|
2021-08-18 14:59:18 +01:00
|
|
|
#define VKCTX(fn) zink_screen(ctx->base.screen)->vk.fn
|
|
|
|
#define VKSCR(fn) screen->vk.fn
|
|
|
|
|
2018-08-31 15:50:20 +01:00
|
|
|
VkFormat
|
2019-10-23 10:59:03 +01:00
|
|
|
zink_get_format(struct zink_screen *screen, enum pipe_format format);
|
2018-08-31 15:50:20 +01:00
|
|
|
|
2021-05-12 16:08:04 +01:00
|
|
|
bool
|
2022-03-25 18:00:21 +00:00
|
|
|
zink_screen_timeline_wait(struct zink_screen *screen, uint64_t batch_id, uint64_t timeout);
|
2021-05-12 16:08:04 +01:00
|
|
|
|
2019-11-06 14:26:12 +00:00
|
|
|
bool
|
|
|
|
zink_is_depth_format_supported(struct zink_screen *screen, VkFormat format);
|
|
|
|
|
2021-06-23 09:23:38 +01:00
|
|
|
#define GET_PROC_ADDR_INSTANCE_LOCAL(screen, instance, x) PFN_vk##x vk_##x = (PFN_vk##x)(screen)->vk_GetInstanceProcAddr(instance, "vk"#x)
|
2020-12-02 22:28:06 +00:00
|
|
|
|
2020-10-06 16:42:27 +01:00
|
|
|
void
|
2021-05-09 16:13:09 +01:00
|
|
|
zink_screen_update_pipeline_cache(struct zink_screen *screen, struct zink_program *pg);
|
|
|
|
|
|
|
|
void
|
|
|
|
zink_screen_get_pipeline_cache(struct zink_screen *screen, struct zink_program *pg);
|
2020-10-06 16:42:27 +01:00
|
|
|
|
2021-04-01 20:18:51 +01:00
|
|
|
void
|
|
|
|
zink_screen_init_descriptor_funcs(struct zink_screen *screen, bool fallback);
|
2021-06-11 17:36:32 +01:00
|
|
|
|
|
|
|
void
|
|
|
|
zink_stub_function_not_loaded(void);
|
2021-08-31 17:23:10 +01:00
|
|
|
|
2022-04-06 19:30:49 +01:00
|
|
|
#define warn_missing_feature(warned, feat) \
|
2021-08-31 17:23:10 +01:00
|
|
|
do { \
|
|
|
|
if (!warned) { \
|
2022-04-06 19:31:26 +01:00
|
|
|
mesa_logw("WARNING: Incorrect rendering will happen " \
|
2021-08-31 17:23:10 +01:00
|
|
|
"because the Vulkan device doesn't support " \
|
2022-04-06 19:31:26 +01:00
|
|
|
"the '%s' feature\n", feat); \
|
2021-08-31 17:23:10 +01:00
|
|
|
warned = true; \
|
|
|
|
} \
|
|
|
|
} while (0)
|
|
|
|
|
2022-05-16 16:27:58 +01:00
|
|
|
#ifdef __cplusplus
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2018-08-31 15:50:20 +01:00
|
|
|
#endif
|