2021-03-02 02:13:17 +00:00
|
|
|
// Project64 - A Nintendo 64 emulator
|
|
|
|
// http://www.pj64-emu.com/
|
|
|
|
// Copyright(C) 2001-2021 Project64
|
|
|
|
// Copyright(C) 2003-2009 Sergey 'Gonetz' Lipski
|
|
|
|
// Copyright(C) 2002 Dave2001
|
|
|
|
// GNU/GPLv2 licensed: https://gnu.org/licenses/gpl-2.0.html
|
2017-04-26 10:23:36 +00:00
|
|
|
#pragma once
|
2013-04-09 12:02:27 +00:00
|
|
|
|
2013-04-04 21:22:19 +00:00
|
|
|
#include <stdio.h>
|
|
|
|
#include <fstream>
|
2016-01-25 10:07:57 +00:00
|
|
|
#include <stdlib.h>
|
2013-04-04 21:22:19 +00:00
|
|
|
#include <stddef.h> // offsetof
|
2013-04-24 05:03:21 +00:00
|
|
|
#include <Common/MemTest.h>
|
2016-02-05 01:48:44 +00:00
|
|
|
#include "Config.h"
|
2016-02-01 08:56:25 +00:00
|
|
|
#include "Settings.h"
|
2021-01-19 05:58:59 +00:00
|
|
|
#include "rdp.h"
|
2013-04-04 21:22:19 +00:00
|
|
|
|
|
|
|
#if defined __VISUALC__
|
|
|
|
#define GLIDE64_TRY __try
|
|
|
|
#define GLIDE64_CATCH __except (EXCEPTION_EXECUTE_HANDLER)
|
|
|
|
#else
|
|
|
|
#define GLIDE64_TRY try
|
|
|
|
#define GLIDE64_CATCH catch (...)
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#if defined(__cplusplus)
|
|
|
|
extern "C" {
|
|
|
|
#endif
|
|
|
|
|
2016-01-20 08:37:38 +00:00
|
|
|
//********
|
|
|
|
// Logging
|
2013-04-04 21:22:19 +00:00
|
|
|
|
2016-01-20 08:37:38 +00:00
|
|
|
// ********************************
|
|
|
|
// ** TAKE OUT BEFORE RELEASE!!! **
|
|
|
|
//#define LOG_UCODE
|
2013-04-04 21:22:19 +00:00
|
|
|
|
2016-01-20 08:37:38 +00:00
|
|
|
// note that some of these things are inserted/removed
|
|
|
|
// from within the code & may not be changed by this define.
|
2013-04-04 21:22:19 +00:00
|
|
|
|
2016-01-20 08:37:38 +00:00
|
|
|
// ********************************
|
2013-04-04 21:22:19 +00:00
|
|
|
|
2016-01-20 08:37:38 +00:00
|
|
|
//#define CATCH_EXCEPTIONS // catch exceptions so it doesn't freeze and will report
|
|
|
|
// "The gfx plugin has caused an exception" instead.
|
2013-04-04 21:22:19 +00:00
|
|
|
|
2016-01-20 08:37:38 +00:00
|
|
|
// Usually enabled
|
2013-04-04 21:22:19 +00:00
|
|
|
#define LARGE_TEXTURE_HANDLING // allow large-textured objects to be split?
|
|
|
|
|
2016-01-20 08:37:38 +00:00
|
|
|
extern unsigned int BMASK;
|
2013-04-04 21:22:19 +00:00
|
|
|
|
2016-01-20 18:33:48 +00:00
|
|
|
extern uint32_t update_screen_count;
|
2013-04-04 21:22:19 +00:00
|
|
|
|
2016-01-20 17:12:15 +00:00
|
|
|
extern int GfxInitDone;
|
2016-03-08 06:31:08 +00:00
|
|
|
extern bool g_romopen;
|
2016-01-20 08:37:38 +00:00
|
|
|
extern int to_fullscreen;
|
2013-04-04 21:22:19 +00:00
|
|
|
|
2016-01-20 08:37:38 +00:00
|
|
|
extern int ev_fullscreen;
|
2013-04-04 21:22:19 +00:00
|
|
|
|
2016-01-20 08:37:38 +00:00
|
|
|
extern int exception;
|
2013-04-04 21:22:19 +00:00
|
|
|
|
2016-01-20 08:37:38 +00:00
|
|
|
int InitGfx();
|
|
|
|
void ReleaseGfx();
|
2013-04-04 21:22:19 +00:00
|
|
|
|
2016-01-20 08:37:38 +00:00
|
|
|
// The highest 8 bits are the segment # (1-16), and the lower 24 bits are the offset to
|
|
|
|
// add to it.
|
2016-01-20 18:33:48 +00:00
|
|
|
__inline uint32_t segoffset(uint32_t so)
|
2016-01-20 08:37:38 +00:00
|
|
|
{
|
|
|
|
return (rdp.segment[(so >> 24) & 0x0f] + (so&BMASK))&BMASK;
|
|
|
|
}
|
2013-04-04 21:22:19 +00:00
|
|
|
|
2016-01-20 08:37:38 +00:00
|
|
|
/* Plugin types */
|
2013-04-04 21:22:19 +00:00
|
|
|
#define PLUGIN_TYPE_GFX 2
|
|
|
|
|
2016-01-25 06:36:56 +00:00
|
|
|
#ifdef _WIN32
|
|
|
|
#define EXPORT extern "C" __declspec(dllexport)
|
|
|
|
#define CALL __cdecl
|
2013-04-04 21:22:19 +00:00
|
|
|
#else
|
2016-01-25 06:36:56 +00:00
|
|
|
#define EXPORT __attribute__((visibility("default")))
|
2013-04-04 21:22:19 +00:00
|
|
|
#define CALL
|
|
|
|
#endif
|
|
|
|
|
2016-01-20 08:37:38 +00:00
|
|
|
/***** Structures *****/
|
2016-01-24 18:41:08 +00:00
|
|
|
typedef struct
|
|
|
|
{
|
|
|
|
uint16_t Version; /* Set to 0x0103 */
|
|
|
|
uint16_t Type; /* Set to PLUGIN_TYPE_GFX */
|
2016-01-20 08:37:38 +00:00
|
|
|
char Name[100]; /* Name of the DLL */
|
|
|
|
|
|
|
|
/* If DLL supports memory these memory options then set them to TRUE or FALSE
|
2016-02-24 07:07:50 +00:00
|
|
|
if it does not support it */
|
2016-01-24 18:45:15 +00:00
|
|
|
int NormalMemory; /* a normal uint8_t array */
|
|
|
|
int MemoryBswaped; /* a normal uint8_t array where the memory has been pre
|
2016-02-24 07:07:50 +00:00
|
|
|
bswap on a dword (32 bits) boundry */
|
2017-05-19 06:23:43 +00:00
|
|
|
} PLUGIN_INFO;
|
2016-01-20 08:37:38 +00:00
|
|
|
|
2016-01-24 18:45:15 +00:00
|
|
|
typedef struct
|
|
|
|
{
|
2017-04-18 12:03:49 +00:00
|
|
|
void * hWnd; /* Render window */
|
|
|
|
void * hStatusBar; /* if render window does not have a status bar then this is NULL */
|
2016-01-20 08:37:38 +00:00
|
|
|
|
2017-04-18 12:03:49 +00:00
|
|
|
int32_t MemoryBswaped; // If this is set to TRUE, then the memory has been pre
|
2016-01-20 08:37:38 +00:00
|
|
|
// bswap on a dword (32 bits) boundry
|
|
|
|
// eg. the first 8 bytes are stored like this:
|
|
|
|
// 4 3 2 1 8 7 6 5
|
|
|
|
|
2016-01-24 18:45:15 +00:00
|
|
|
uint8_t * HEADER; // This is the rom header (first 40h bytes of the rom
|
2016-01-20 08:37:38 +00:00
|
|
|
// This will be in the same memory format as the rest of the memory.
|
2016-01-24 18:45:15 +00:00
|
|
|
uint8_t * RDRAM;
|
|
|
|
uint8_t * DMEM;
|
|
|
|
uint8_t * IMEM;
|
2016-01-20 08:37:38 +00:00
|
|
|
|
2016-01-20 18:33:48 +00:00
|
|
|
uint32_t * MI_INTR_REG;
|
|
|
|
|
|
|
|
uint32_t * DPC_START_REG;
|
|
|
|
uint32_t * DPC_END_REG;
|
|
|
|
uint32_t * DPC_CURRENT_REG;
|
|
|
|
uint32_t * DPC_STATUS_REG;
|
|
|
|
uint32_t * DPC_CLOCK_REG;
|
|
|
|
uint32_t * DPC_BUFBUSY_REG;
|
|
|
|
uint32_t * DPC_PIPEBUSY_REG;
|
|
|
|
uint32_t * DPC_TMEM_REG;
|
|
|
|
|
|
|
|
uint32_t * VI_STATUS_REG;
|
|
|
|
uint32_t * VI_ORIGIN_REG;
|
|
|
|
uint32_t * VI_WIDTH_REG;
|
|
|
|
uint32_t * VI_INTR_REG;
|
|
|
|
uint32_t * VI_V_CURRENT_LINE_REG;
|
|
|
|
uint32_t * VI_TIMING_REG;
|
|
|
|
uint32_t * VI_V_SYNC_REG;
|
|
|
|
uint32_t * VI_H_SYNC_REG;
|
|
|
|
uint32_t * VI_LEAP_REG;
|
|
|
|
uint32_t * VI_H_START_REG;
|
|
|
|
uint32_t * VI_V_START_REG;
|
|
|
|
uint32_t * VI_V_BURST_REG;
|
|
|
|
uint32_t * VI_X_SCALE_REG;
|
|
|
|
uint32_t * VI_Y_SCALE_REG;
|
2016-01-20 08:37:38 +00:00
|
|
|
|
|
|
|
void(*CheckInterrupts)(void);
|
2016-03-10 18:29:41 +00:00
|
|
|
#ifdef ANDROID
|
|
|
|
void(CALL *SwapBuffers)(void);
|
|
|
|
#endif
|
2016-01-20 08:37:38 +00:00
|
|
|
} GFX_INFO;
|
|
|
|
|
|
|
|
extern GFX_INFO gfx;
|
|
|
|
extern bool no_dlist;
|
|
|
|
|
|
|
|
/******************************************************************
|
2016-02-24 07:07:50 +00:00
|
|
|
Function: CaptureScreen
|
|
|
|
Purpose: This function dumps the current frame to a file
|
|
|
|
input: pointer to the directory to save the file to
|
|
|
|
output: none
|
|
|
|
*******************************************************************/
|
2016-01-20 08:37:38 +00:00
|
|
|
EXPORT void CALL CaptureScreen(char * Directory);
|
|
|
|
|
|
|
|
/******************************************************************
|
2016-02-24 07:07:50 +00:00
|
|
|
Function: ChangeWindow
|
|
|
|
Purpose: to change the window between fullscreen and window
|
|
|
|
mode. If the window was in fullscreen this should
|
|
|
|
change the screen to window mode and vice vesa.
|
|
|
|
input: none
|
|
|
|
output: none
|
|
|
|
*******************************************************************/
|
2016-01-20 08:37:38 +00:00
|
|
|
EXPORT void CALL ChangeWindow(void);
|
|
|
|
|
|
|
|
/******************************************************************
|
2016-02-24 07:07:50 +00:00
|
|
|
Function: CloseDLL
|
|
|
|
Purpose: This function is called when the emulator is closing
|
|
|
|
down allowing the dll to de-initialise.
|
|
|
|
input: none
|
|
|
|
output: none
|
|
|
|
*******************************************************************/
|
2016-01-20 08:37:38 +00:00
|
|
|
EXPORT void CALL CloseDLL(void);
|
|
|
|
|
|
|
|
/******************************************************************
|
2016-02-24 07:07:50 +00:00
|
|
|
Function: DllAbout
|
|
|
|
Purpose: This function is optional function that is provided
|
|
|
|
to give further information about the DLL.
|
|
|
|
input: a handle to the window that calls this function
|
|
|
|
output: none
|
|
|
|
*******************************************************************/
|
2017-04-18 12:03:49 +00:00
|
|
|
EXPORT void CALL DllAbout(void * hParent);
|
2016-01-20 08:37:38 +00:00
|
|
|
|
|
|
|
/******************************************************************
|
2016-02-24 07:07:50 +00:00
|
|
|
Function: DllConfig
|
|
|
|
Purpose: This function is optional function that is provided
|
|
|
|
to allow the user to configure the dll
|
|
|
|
input: a handle to the window that calls this function
|
|
|
|
output: none
|
|
|
|
*******************************************************************/
|
2017-04-18 12:03:49 +00:00
|
|
|
EXPORT void CALL DllConfig(void * hParent);
|
2016-01-20 08:37:38 +00:00
|
|
|
|
|
|
|
/******************************************************************
|
2016-02-24 07:07:50 +00:00
|
|
|
Function: DllTest
|
|
|
|
Purpose: This function is optional function that is provided
|
|
|
|
to allow the user to test the dll
|
|
|
|
input: a handle to the window that calls this function
|
|
|
|
output: none
|
|
|
|
*******************************************************************/
|
2017-04-18 12:03:49 +00:00
|
|
|
EXPORT void CALL DllTest(void * hParent);
|
2016-01-20 08:37:38 +00:00
|
|
|
|
|
|
|
/******************************************************************
|
2016-02-24 07:07:50 +00:00
|
|
|
Function: DrawScreen
|
|
|
|
Purpose: This function is called when the emulator receives a
|
|
|
|
WM_PAINT message. This allows the gfx to fit in when
|
|
|
|
it is being used in the desktop.
|
|
|
|
input: none
|
|
|
|
output: none
|
|
|
|
*******************************************************************/
|
2016-01-20 08:37:38 +00:00
|
|
|
EXPORT void CALL DrawScreen(void);
|
|
|
|
|
|
|
|
/******************************************************************
|
2016-02-24 07:07:50 +00:00
|
|
|
Function: GetDllInfo
|
|
|
|
Purpose: This function allows the emulator to gather information
|
|
|
|
about the dll by filling in the PluginInfo structure.
|
|
|
|
input: a pointer to a PLUGIN_INFO stucture that needs to be
|
|
|
|
filled by the function. (see def above)
|
|
|
|
output: none
|
|
|
|
*******************************************************************/
|
2016-01-20 08:37:38 +00:00
|
|
|
EXPORT void CALL GetDllInfo(PLUGIN_INFO * PluginInfo);
|
|
|
|
|
|
|
|
/******************************************************************
|
2016-02-24 07:07:50 +00:00
|
|
|
Function: InitiateGFX
|
|
|
|
Purpose: This function is called when the DLL is started to give
|
|
|
|
information from the emulator that the n64 graphics
|
|
|
|
uses. This is not called from the emulation thread.
|
|
|
|
Input: Gfx_Info is passed to this function which is defined
|
|
|
|
above.
|
|
|
|
Output: TRUE on success
|
|
|
|
FALSE on failure to initialise
|
|
|
|
|
|
|
|
** note on interrupts **:
|
|
|
|
To generate an interrupt set the appropriate bit in MI_INTR_REG
|
|
|
|
and then call the function CheckInterrupts to tell the emulator
|
|
|
|
that there is a waiting interrupt.
|
|
|
|
*******************************************************************/
|
2016-01-20 08:37:38 +00:00
|
|
|
EXPORT int CALL InitiateGFX(GFX_INFO Gfx_Info);
|
|
|
|
|
|
|
|
/******************************************************************
|
2016-02-24 07:07:50 +00:00
|
|
|
Function: MoveScreen
|
|
|
|
Purpose: This function is called in response to the emulator
|
|
|
|
receiving a WM_MOVE passing the xpos and ypos passed
|
|
|
|
from that message.
|
|
|
|
input: xpos - the x-coordinate of the upper-left corner of the
|
|
|
|
client area of the window.
|
|
|
|
ypos - y-coordinate of the upper-left corner of the
|
|
|
|
client area of the window.
|
|
|
|
output: none
|
|
|
|
*******************************************************************/
|
2016-01-20 08:37:38 +00:00
|
|
|
EXPORT void CALL MoveScreen(int xpos, int ypos);
|
|
|
|
|
|
|
|
/******************************************************************
|
2016-02-24 07:07:50 +00:00
|
|
|
Function: ProcessDList
|
|
|
|
Purpose: This function is called when there is a Dlist to be
|
|
|
|
processed. (High level GFX list)
|
|
|
|
input: none
|
|
|
|
output: none
|
|
|
|
*******************************************************************/
|
2016-01-20 08:37:38 +00:00
|
|
|
EXPORT void CALL ProcessDList(void);
|
|
|
|
|
|
|
|
/******************************************************************
|
2016-02-24 07:07:50 +00:00
|
|
|
Function: ProcessRDPList
|
|
|
|
Purpose: This function is called when there is a Dlist to be
|
|
|
|
processed. (Low level GFX list)
|
|
|
|
input: none
|
|
|
|
output: none
|
|
|
|
*******************************************************************/
|
2016-01-20 08:37:38 +00:00
|
|
|
EXPORT void CALL ProcessRDPList(void);
|
|
|
|
|
|
|
|
/******************************************************************
|
2016-02-24 07:07:50 +00:00
|
|
|
Function: RomClosed
|
|
|
|
Purpose: This function is called when a rom is closed.
|
|
|
|
input: none
|
|
|
|
output: none
|
|
|
|
*******************************************************************/
|
2016-01-20 08:37:38 +00:00
|
|
|
EXPORT void CALL RomClosed(void);
|
|
|
|
|
|
|
|
/******************************************************************
|
2016-02-24 07:07:50 +00:00
|
|
|
Function: RomOpen
|
|
|
|
Purpose: This function is called when a rom is open. (from the
|
|
|
|
emulation thread)
|
|
|
|
input: none
|
|
|
|
output: none
|
|
|
|
*******************************************************************/
|
2016-01-20 08:37:38 +00:00
|
|
|
EXPORT void CALL RomOpen(void);
|
|
|
|
|
|
|
|
/******************************************************************
|
2016-02-24 07:07:50 +00:00
|
|
|
Function: ShowCFB
|
|
|
|
Purpose: Useally once Dlists are started being displayed, cfb is
|
|
|
|
ignored. This function tells the dll to start displaying
|
|
|
|
them again.
|
|
|
|
input: none
|
|
|
|
output: none
|
|
|
|
*******************************************************************/
|
2016-01-20 08:37:38 +00:00
|
|
|
EXPORT void CALL ShowCFB(void);
|
|
|
|
|
|
|
|
/******************************************************************
|
2016-02-24 07:07:50 +00:00
|
|
|
Function: UpdateScreen
|
|
|
|
Purpose: This function is called in response to a vsync of the
|
|
|
|
screen were the VI bit in MI_INTR_REG has already been
|
|
|
|
set
|
|
|
|
input: none
|
|
|
|
output: none
|
|
|
|
*******************************************************************/
|
2016-01-20 08:37:38 +00:00
|
|
|
EXPORT void CALL UpdateScreen(void);
|
|
|
|
|
|
|
|
/******************************************************************
|
2016-02-24 07:07:50 +00:00
|
|
|
Function: ViStatusChanged
|
|
|
|
Purpose: This function is called to notify the dll that the
|
|
|
|
ViStatus registers value has been changed.
|
|
|
|
input: none
|
|
|
|
output: none
|
|
|
|
*******************************************************************/
|
2016-01-20 08:37:38 +00:00
|
|
|
EXPORT void CALL ViStatusChanged(void);
|
|
|
|
|
|
|
|
/******************************************************************
|
2016-02-24 07:07:50 +00:00
|
|
|
Function: ViWidthChanged
|
|
|
|
Purpose: This function is called to notify the dll that the
|
|
|
|
ViWidth registers value has been changed.
|
|
|
|
input: none
|
|
|
|
output: none
|
|
|
|
*******************************************************************/
|
2016-01-20 08:37:38 +00:00
|
|
|
EXPORT void CALL ViWidthChanged(void);
|
|
|
|
|
2016-03-10 18:29:41 +00:00
|
|
|
#ifdef ANDROID
|
|
|
|
/******************************************************************
|
|
|
|
Function: SurfaceCreated
|
|
|
|
Purpose: this function is called when the surface is created.
|
|
|
|
input: none
|
|
|
|
output: none
|
|
|
|
*******************************************************************/
|
|
|
|
EXPORT void CALL SurfaceCreated(void);
|
|
|
|
/******************************************************************
|
|
|
|
Function: SurfaceChanged
|
|
|
|
Purpose: this function is called when the surface is has changed.
|
|
|
|
input: none
|
|
|
|
output: none
|
|
|
|
*******************************************************************/
|
|
|
|
EXPORT void CALL SurfaceChanged(int width, int height);
|
|
|
|
#endif
|
2016-01-20 08:37:38 +00:00
|
|
|
/******************************************************************
|
2016-02-24 07:07:50 +00:00
|
|
|
Function: FrameBufferWrite
|
|
|
|
Purpose: This function is called to notify the dll that the
|
|
|
|
frame buffer has been modified by CPU at the given address.
|
|
|
|
input: addr rdram address
|
|
|
|
val val
|
|
|
|
size 1 = uint8_t, 2 = uint16_t, 4 = uint32_t
|
|
|
|
output: none
|
|
|
|
*******************************************************************/
|
2016-01-20 18:33:48 +00:00
|
|
|
EXPORT void CALL FBWrite(uint32_t, uint32_t);
|
2016-01-20 08:37:38 +00:00
|
|
|
|
|
|
|
typedef struct
|
|
|
|
{
|
2016-01-20 18:33:48 +00:00
|
|
|
uint32_t addr;
|
|
|
|
uint32_t val;
|
2016-01-24 18:45:15 +00:00
|
|
|
uint32_t size; // 1 = uint8_t, 2 = uint16_t, 4=uint32_t
|
2016-01-20 08:37:38 +00:00
|
|
|
} FrameBufferModifyEntry;
|
|
|
|
|
|
|
|
/******************************************************************
|
2016-02-24 07:07:50 +00:00
|
|
|
Function: FrameBufferWriteList
|
|
|
|
Purpose: This function is called to notify the dll that the
|
|
|
|
frame buffer has been modified by CPU at the given address.
|
|
|
|
input: FrameBufferModifyEntry *plist
|
|
|
|
size = size of the plist, max = 1024
|
|
|
|
output: none
|
|
|
|
*******************************************************************/
|
2016-01-20 18:33:48 +00:00
|
|
|
EXPORT void CALL FBWList(FrameBufferModifyEntry *plist, uint32_t size);
|
2016-01-20 08:37:38 +00:00
|
|
|
|
|
|
|
/******************************************************************
|
2016-02-24 07:07:50 +00:00
|
|
|
Function: FrameBufferRead
|
|
|
|
Purpose: This function is called to notify the dll that the
|
|
|
|
frame buffer memory is beening read at the given address.
|
|
|
|
DLL should copy content from its render buffer to the frame buffer
|
|
|
|
in N64 RDRAM
|
|
|
|
DLL is responsible to maintain its own frame buffer memory addr list
|
|
|
|
DLL should copy 4KB block content back to RDRAM frame buffer.
|
|
|
|
Emulator should not call this function again if other memory
|
|
|
|
is read within the same 4KB range
|
|
|
|
input: addr rdram address
|
|
|
|
val val
|
|
|
|
size 1 = uint8_t, 2 = uint16_t, 4 = uint32_t
|
|
|
|
output: none
|
|
|
|
*******************************************************************/
|
2016-01-20 18:33:48 +00:00
|
|
|
EXPORT void CALL FBRead(uint32_t addr);
|
2016-01-20 08:37:38 +00:00
|
|
|
|
|
|
|
/************************************************************************
|
|
|
|
Function: FBGetFrameBufferInfo
|
|
|
|
Purpose: This function is called by the emulator core to retrieve depth
|
|
|
|
buffer information from the video plugin in order to be able
|
|
|
|
to notify the video plugin about CPU depth buffer read/write
|
|
|
|
operations
|
|
|
|
|
|
|
|
size:
|
|
|
|
= 1 byte
|
|
|
|
= 2 word (16 bit) <-- this is N64 default depth buffer format
|
|
|
|
= 4 dword (32 bit)
|
|
|
|
|
|
|
|
when depth buffer information is not available yet, set all values
|
|
|
|
in the FrameBufferInfo structure to 0
|
|
|
|
|
|
|
|
input: FrameBufferInfo *pinfo
|
|
|
|
pinfo is pointed to a FrameBufferInfo structure which to be
|
|
|
|
filled in by this function
|
|
|
|
output: Values are return in the FrameBufferInfo structure
|
|
|
|
************************************************************************/
|
|
|
|
EXPORT void CALL FBGetFrameBufferInfo(void *pinfo);
|
|
|
|
|
|
|
|
EXPORT void CALL PluginLoaded(void);
|
2013-04-04 21:22:19 +00:00
|
|
|
|
|
|
|
#if defined(__cplusplus)
|
2017-05-21 10:44:34 +00:00
|
|
|
}
|
2013-04-04 21:22:19 +00:00
|
|
|
#endif
|