/* Created by Language version: 6.2.0 */
/* VECTORIZED */
#include <stdio.h>
#include <stdlib.h>
#include <math.h>
#include "scoplib_ansi.h"
#undef PI
#define nil 0
#include "md1redef.h"
#include "section.h"
#include "nrniv_mf.h"
#include "md2redef.h"
#if METHOD3
extern int _method3;
#endif
#if !NRNGPU
#undef exp
#define exp hoc_Exp
extern double hoc_Exp(double);
#endif
#define _threadargscomma_ _p, _ppvar, _thread, _nt,
#define _threadargs_ _p, _ppvar, _thread, _nt
#define _threadargsprotocomma_ double* _p, Datum* _ppvar, Datum* _thread, _NrnThread* _nt,
#define _threadargsproto_ double* _p, Datum* _ppvar, Datum* _thread, _NrnThread* _nt
/*SUPPRESS 761*/
/*SUPPRESS 762*/
/*SUPPRESS 763*/
/*SUPPRESS 765*/
extern double *getarg();
/* Thread safe. No static _p or _ppvar. */
#define t _nt->_t
#define dt _nt->_dt
#define tau_r_AMPA _p[0]
#define tau_d_AMPA _p[1]
#define tau_r_NMDA _p[2]
#define tau_d_NMDA _p[3]
#define Use _p[4]
#define Dep _p[5]
#define Fac _p[6]
#define e _p[7]
#define mg _p[8]
#define u0 _p[9]
#define synapseID _p[10]
#define verboseLevel _p[11]
#define NMDA_ratio _p[12]
#define i _p[13]
#define i_AMPA _p[14]
#define i_NMDA _p[15]
#define g_AMPA _p[16]
#define g_NMDA _p[17]
#define g _p[18]
#define A_AMPA_step _p[19]
#define B_AMPA_step _p[20]
#define A_NMDA_step _p[21]
#define B_NMDA_step _p[22]
#define Rstate _p[23]
#define tsyn_fac _p[24]
#define u _p[25]
#define A_AMPA _p[26]
#define B_AMPA _p[27]
#define A_NMDA _p[28]
#define B_NMDA _p[29]
#define factor_AMPA _p[30]
#define factor_NMDA _p[31]
#define DA_AMPA _p[32]
#define DB_AMPA _p[33]
#define DA_NMDA _p[34]
#define DB_NMDA _p[35]
#define v _p[36]
#define _g _p[37]
#define _tsav _p[38]
#define _nd_area *_ppvar[0]._pval
#define rng *_ppvar[2]._pval
#define _p_rng _ppvar[2]._pval
#if MAC
#if !defined(v)
#define v _mlhv
#endif
#if !defined(h)
#define h _mlhh
#endif
#endif
#if defined(__cplusplus)
extern "C" {
#endif
static int hoc_nrnpointerindex = 2;
static Datum* _extcall_thread;
static Prop* _extcall_prop;
/* external NEURON variables */
/* declaration of user functions */
static double _hoc_setRNG();
static double _hoc_state();
static double _hoc_toggleVerbose();
static double _hoc_urand();
static int _mechtype;
extern void _nrn_cacheloop_reg(int, int);
extern void hoc_register_prop_size(int, int, int);
extern void hoc_register_limits(int, HocParmLimits*);
extern void hoc_register_units(int, HocParmUnits*);
extern void nrn_promote(Prop*, int, int);
extern Memb_func* memb_func;
extern Prop* nrn_point_prop_;
static int _pointtype;
static void* _hoc_create_pnt(_ho) Object* _ho; { void* create_point_process();
return create_point_process(_pointtype, _ho);
}
static void _hoc_destroy_pnt();
static double _hoc_loc_pnt(_vptr) void* _vptr; {double loc_point_process();
return loc_point_process(_pointtype, _vptr);
}
static double _hoc_has_loc(_vptr) void* _vptr; {double has_loc_point();
return has_loc_point(_vptr);
}
static double _hoc_get_loc_pnt(_vptr)void* _vptr; {
double get_loc_point_process(); return (get_loc_point_process(_vptr));
}
extern void _nrn_setdata_reg(int, void(*)(Prop*));
static void _setdata(Prop* _prop) {
_extcall_prop = _prop;
}
static void _hoc_setdata(void* _vptr) { Prop* _prop;
_prop = ((Point_process*)_vptr)->_prop;
_setdata(_prop);
}
/* connect user functions to hoc names */
static VoidFunc hoc_intfunc[] = {
0,0
};
static Member_func _member_func[] = {
"loc", _hoc_loc_pnt,
"has_loc", _hoc_has_loc,
"get_loc", _hoc_get_loc_pnt,
"setRNG", _hoc_setRNG,
"state", _hoc_state,
"toggleVerbose", _hoc_toggleVerbose,
"urand", _hoc_urand,
0, 0
};
#define toggleVerbose toggleVerbose_ProbAMPANMDA_EMS
#define urand urand_ProbAMPANMDA_EMS
extern double toggleVerbose( _threadargsproto_ );
extern double urand( _threadargsproto_ );
/* declare global and static user variables */
static int _thread1data_inuse = 0;
static double _thread1data[1];
#define _gth 0
#define gmax gmax_ProbAMPANMDA_EMS
double gmax = 0.001;
#define mggate_ProbAMPANMDA_EMS _thread1data[0]
#define mggate _thread[_gth]._pval[0]
/* some parameters have upper and lower limits */
static HocParmLimits _hoc_parm_limits[] = {
0,0,0
};
static HocParmUnits _hoc_parm_units[] = {
"gmax_ProbAMPANMDA_EMS", "uS",
"tau_r_AMPA", "ms",
"tau_d_AMPA", "ms",
"tau_r_NMDA", "ms",
"tau_d_NMDA", "ms",
"Use", "1",
"Dep", "ms",
"Fac", "ms",
"e", "mV",
"mg", "mM",
"NMDA_ratio", "1",
"i", "nA",
"i_AMPA", "nA",
"i_NMDA", "nA",
"g_AMPA", "uS",
"g_NMDA", "uS",
"g", "uS",
"Rstate", "1",
"tsyn_fac", "ms",
"u", "1",
0,0
};
static double A_NMDA0 = 0;
static double A_AMPA0 = 0;
static double B_NMDA0 = 0;
static double B_AMPA0 = 0;
static double delta_t = 0.01;
/* connect global user variables to hoc */
static DoubScal hoc_scdoub[] = {
"mggate_ProbAMPANMDA_EMS", &mggate_ProbAMPANMDA_EMS,
"gmax_ProbAMPANMDA_EMS", &gmax_ProbAMPANMDA_EMS,
0,0
};
static DoubVec hoc_vdoub[] = {
0,0,0
};
static double _sav_indep;
static void nrn_alloc(Prop*);
static void nrn_init(_NrnThread*, _Memb_list*, int);
static void nrn_state(_NrnThread*, _Memb_list*, int);
static void nrn_cur(_NrnThread*, _Memb_list*, int);
static void nrn_jacob(_NrnThread*, _Memb_list*, int);
static void _hoc_destroy_pnt(_vptr) void* _vptr; {
destroy_point_process(_vptr);
}
static int _ode_count(int);
/* connect range variables in _p that hoc is supposed to know about */
static const char *_mechanism[] = {
"6.2.0",
"ProbAMPANMDA_EMS",
"tau_r_AMPA",
"tau_d_AMPA",
"tau_r_NMDA",
"tau_d_NMDA",
"Use",
"Dep",
"Fac",
"e",
"mg",
"u0",
"synapseID",
"verboseLevel",
"NMDA_ratio",
0,
"i",
"i_AMPA",
"i_NMDA",
"g_AMPA",
"g_NMDA",
"g",
"A_AMPA_step",
"B_AMPA_step",
"A_NMDA_step",
"B_NMDA_step",
"Rstate",
"tsyn_fac",
"u",
0,
"A_AMPA",
"B_AMPA",
"A_NMDA",
"B_NMDA",
0,
"rng",
0};
extern Prop* need_memb(Symbol*);
static void nrn_alloc(Prop* _prop) {
Prop *prop_ion;
double *_p; Datum *_ppvar;
if (nrn_point_prop_) {
_prop->_alloc_seq = nrn_point_prop_->_alloc_seq;
_p = nrn_point_prop_->param;
_ppvar = nrn_point_prop_->dparam;
}else{
_p = nrn_prop_data_alloc(_mechtype, 39, _prop);
/*initialize range parameters*/
tau_r_AMPA = 0.2;
tau_d_AMPA = 1.7;
tau_r_NMDA = 0.29;
tau_d_NMDA = 43;
Use = 1;
Dep = 100;
Fac = 10;
e = 0;
mg = 1;
u0 = 0;
synapseID = 0;
verboseLevel = 0;
NMDA_ratio = 0.71;
}
_prop->param = _p;
_prop->param_size = 39;
if (!nrn_point_prop_) {
_ppvar = nrn_prop_datum_alloc(_mechtype, 3, _prop);
}
_prop->dparam = _ppvar;
/*connect ionic variables to this model*/
}
static void _initlists();
static void _net_receive(Point_process*, double*, double);
static void _net_init(Point_process*, double*, double);
static void _thread_mem_init(Datum*);
static void _thread_cleanup(Datum*);
extern Symbol* hoc_lookup(const char*);
extern void _nrn_thread_reg(int, int, void(*f)(Datum*));
extern void _nrn_thread_table_reg(int, void(*)(double*, Datum*, Datum*, _NrnThread*, int));
extern void hoc_register_tolerance(int, HocStateTolerance*, Symbol***);
extern void _cvode_abstol( Symbol**, double*, int);
void _ProbAMPANMDA_EMS_reg() {
int _vectorized = 1;
_initlists();
_pointtype = point_register_mech(_mechanism,
nrn_alloc,nrn_cur, nrn_jacob, nrn_state, nrn_init,
hoc_nrnpointerindex, 2,
_hoc_create_pnt, _hoc_destroy_pnt, _member_func);
_extcall_thread = (Datum*)ecalloc(1, sizeof(Datum));
_thread_mem_init(_extcall_thread);
_thread1data_inuse = 0;
_mechtype = nrn_get_mechtype(_mechanism[1]);
_nrn_setdata_reg(_mechtype, _setdata);
_nrn_thread_reg(_mechtype, 1, _thread_mem_init);
_nrn_thread_reg(_mechtype, 0, _thread_cleanup);
hoc_register_prop_size(_mechtype, 39, 3);
hoc_register_cvode(_mechtype, _ode_count, 0, 0, 0);
pnt_receive[_mechtype] = _net_receive;
pnt_receive_init[_mechtype] = _net_init;
pnt_receive_size[_mechtype] = 5;
hoc_register_var(hoc_scdoub, hoc_vdoub, hoc_intfunc);
ivoc_help("help ?1 ProbAMPANMDA_EMS C:/Users/Fabio/Desktop/NEURON_Book/Markram/hoc_combos_syn.1_0_10.allzips/Neurons/Fabio_L5_TTPC1_cADpyr232_5/mechanisms/ProbAMPANMDA_EMS.mod\n");
hoc_register_limits(_mechtype, _hoc_parm_limits);
hoc_register_units(_mechtype, _hoc_parm_units);
}
static int _reset;
static char *modelname = "Probabilistic AMPA and NMDA receptor with presynaptic short-term plasticity ";
static int error;
static int _ninits = 0;
static int _match_recurse=1;
static void _modl_cleanup(){ _match_recurse=1;}
static int setRNG(_threadargsproto_);
static int state(_threadargsproto_);
/*VERBATIM*/
#include<stdlib.h>
#include<stdio.h>
#include<math.h>
double nrn_random_pick(void* r);
void* nrn_random_arg(int argpos);
static int state ( _threadargsproto_ ) {
A_AMPA = A_AMPA * A_AMPA_step ;
B_AMPA = B_AMPA * B_AMPA_step ;
A_NMDA = A_NMDA * A_NMDA_step ;
B_NMDA = B_NMDA * B_NMDA_step ;
return 0; }
static double _hoc_state(void* _vptr) {
double _r;
double* _p; Datum* _ppvar; Datum* _thread; _NrnThread* _nt;
_p = ((Point_process*)_vptr)->_prop->param;
_ppvar = ((Point_process*)_vptr)->_prop->dparam;
_thread = _extcall_thread;
_nt = (_NrnThread*)((Point_process*)_vptr)->_vnt;
_r = 1.;
state ( _p, _ppvar, _thread, _nt );
return(_r);
}
static void _net_receive (_pnt, _args, _lflag) Point_process* _pnt; double* _args; double _lflag;
{ double* _p; Datum* _ppvar; Datum* _thread; _NrnThread* _nt;
_thread = (Datum*)0; _nt = (_NrnThread*)_pnt->_vnt; _p = _pnt->_prop->param; _ppvar = _pnt->_prop->dparam;
if (_tsav > t){ extern char* hoc_object_name(); hoc_execerror(hoc_object_name(_pnt->ob), ":Event arrived out of order. Must call ParallelContext.set_maxstep AFTER assigning minimum NetCon.delay");}
_tsav = t; {
double _lresult ;
_args[1] = _args[0] ;
_args[2] = _args[0] * NMDA_ratio ;
if ( ! ( _args[0] > 0.0 ) ) {
/*VERBATIM*/
return;
}
if ( Fac > 0.0 ) {
u = u * exp ( - ( t - tsyn_fac ) / Fac ) ;
}
else {
u = Use ;
}
if ( Fac > 0.0 ) {
u = u + Use * ( 1.0 - u ) ;
}
tsyn_fac = t ;
if ( Rstate == 0.0 ) {
_args[3] = exp ( - ( t - _args[4] ) / Dep ) ;
_lresult = urand ( _threadargs_ ) ;
if ( _lresult > _args[3] ) {
Rstate = 1.0 ;
if ( verboseLevel > 0.0 ) {
printf ( "Recovered! %f at time %g: Psurv = %g, urand=%g\n" , synapseID , t , _args[3] , _lresult ) ;
}
}
else {
_args[4] = t ;
if ( verboseLevel > 0.0 ) {
printf ( "Failed to recover! %f at time %g: Psurv = %g, urand=%g\n" , synapseID , t , _args[3] , _lresult ) ;
}
}
}
if ( Rstate == 1.0 ) {
_lresult = urand ( _threadargs_ ) ;
if ( _lresult < u ) {
_args[4] = t ;
Rstate = 0.0 ;
A_AMPA = A_AMPA + _args[1] * factor_AMPA ;
B_AMPA = B_AMPA + _args[1] * factor_AMPA ;
A_NMDA = A_NMDA + _args[2] * factor_NMDA ;
B_NMDA = B_NMDA + _args[2] * factor_NMDA ;
if ( verboseLevel > 0.0 ) {
printf ( "Release! %f at time %g: vals %g %g %g %g\n" , synapseID , t , A_AMPA , _args[1] , factor_AMPA , _args[0] ) ;
}
}
else {
if ( verboseLevel > 0.0 ) {
printf ( "Failure! %f at time %g: urand = %g\n" , synapseID , t , _lresult ) ;
}
}
}
} }
static void _net_init(Point_process* _pnt, double* _args, double _lflag) {
double* _p = _pnt->_prop->param;
Datum* _ppvar = _pnt->_prop->dparam;
Datum* _thread = (Datum*)0;
_NrnThread* _nt = (_NrnThread*)_pnt->_vnt;
_args[4] = t ;
}
static int setRNG ( _threadargsproto_ ) {
/*VERBATIM*/
{
/**
* This function takes a NEURON Random object declared in hoc and makes it usable by this mod file.
* Note that this method is taken from Brett paper as used by netstim.hoc and netstim.mod
* which points out that the Random must be in uniform(1) mode
*/
void** pv = (void**)(&_p_rng);
if( ifarg(1)) {
*pv = nrn_random_arg(1);
} else {
*pv = (void*)0;
}
}
return 0; }
static double _hoc_setRNG(void* _vptr) {
double _r;
double* _p; Datum* _ppvar; Datum* _thread; _NrnThread* _nt;
_p = ((Point_process*)_vptr)->_prop->param;
_ppvar = ((Point_process*)_vptr)->_prop->dparam;
_thread = _extcall_thread;
_nt = (_NrnThread*)((Point_process*)_vptr)->_vnt;
_r = 1.;
setRNG ( _p, _ppvar, _thread, _nt );
return(_r);
}
double urand ( _threadargsproto_ ) {
double _lurand;
/*VERBATIM*/
double value;
if (_p_rng) {
/*
:Supports separate independent but reproducible streams for
: each instance. However, the corresponding hoc Random
: distribution MUST be set to Random.negexp(1)
*/
value = nrn_random_pick(_p_rng);
//printf("random stream for this simulation = %lf\n",value);
return value;
}else{
value = scop_random ( 1.0 ) ;
/*VERBATIM*/
}
_lurand = value ;
return _lurand;
}
static double _hoc_urand(void* _vptr) {
double _r;
double* _p; Datum* _ppvar; Datum* _thread; _NrnThread* _nt;
_p = ((Point_process*)_vptr)->_prop->param;
_ppvar = ((Point_process*)_vptr)->_prop->dparam;
_thread = _extcall_thread;
_nt = (_NrnThread*)((Point_process*)_vptr)->_vnt;
_r = urand ( _p, _ppvar, _thread, _nt );
return(_r);
}
double toggleVerbose ( _threadargsproto_ ) {
double _ltoggleVerbose;
verboseLevel = 1.0 - verboseLevel ;
return _ltoggleVerbose;
}
static double _hoc_toggleVerbose(void* _vptr) {
double _r;
double* _p; Datum* _ppvar; Datum* _thread; _NrnThread* _nt;
_p = ((Point_process*)_vptr)->_prop->param;
_ppvar = ((Point_process*)_vptr)->_prop->dparam;
_thread = _extcall_thread;
_nt = (_NrnThread*)((Point_process*)_vptr)->_vnt;
_r = toggleVerbose ( _p, _ppvar, _thread, _nt );
return(_r);
}
static int _ode_count(int _type){ hoc_execerror("ProbAMPANMDA_EMS", "cannot be used with CVODE"); return 0;}
static void _thread_mem_init(Datum* _thread) {
if (_thread1data_inuse) {_thread[_gth]._pval = (double*)ecalloc(1, sizeof(double));
}else{
_thread[_gth]._pval = _thread1data; _thread1data_inuse = 1;
}
}
static void _thread_cleanup(Datum* _thread) {
if (_thread[_gth]._pval == _thread1data) {
_thread1data_inuse = 0;
}else{
free((void*)_thread[_gth]._pval);
}
}
static void initmodel(double* _p, Datum* _ppvar, Datum* _thread, _NrnThread* _nt) {
int _i; double _save;{
A_NMDA = A_NMDA0;
A_AMPA = A_AMPA0;
B_NMDA = B_NMDA0;
B_AMPA = B_AMPA0;
{
double _ltp_AMPA , _ltp_NMDA ;
Rstate = 1.0 ;
tsyn_fac = 0.0 ;
u = u0 ;
A_AMPA = 0.0 ;
B_AMPA = 0.0 ;
A_NMDA = 0.0 ;
B_NMDA = 0.0 ;
_ltp_AMPA = ( tau_r_AMPA * tau_d_AMPA ) / ( tau_d_AMPA - tau_r_AMPA ) * log ( tau_d_AMPA / tau_r_AMPA ) ;
_ltp_NMDA = ( tau_r_NMDA * tau_d_NMDA ) / ( tau_d_NMDA - tau_r_NMDA ) * log ( tau_d_NMDA / tau_r_NMDA ) ;
factor_AMPA = - exp ( - _ltp_AMPA / tau_r_AMPA ) + exp ( - _ltp_AMPA / tau_d_AMPA ) ;
factor_AMPA = 1.0 / factor_AMPA ;
factor_NMDA = - exp ( - _ltp_NMDA / tau_r_NMDA ) + exp ( - _ltp_NMDA / tau_d_NMDA ) ;
factor_NMDA = 1.0 / factor_NMDA ;
A_AMPA_step = exp ( dt * ( ( - 1.0 ) / tau_r_AMPA ) ) ;
B_AMPA_step = exp ( dt * ( ( - 1.0 ) / tau_d_AMPA ) ) ;
A_NMDA_step = exp ( dt * ( ( - 1.0 ) / tau_r_NMDA ) ) ;
B_NMDA_step = exp ( dt * ( ( - 1.0 ) / tau_d_NMDA ) ) ;
}
}
}
static void nrn_init(_NrnThread* _nt, _Memb_list* _ml, int _type){
double* _p; Datum* _ppvar; Datum* _thread;
Node *_nd; double _v; int* _ni; int _iml, _cntml;
#if CACHEVEC
_ni = _ml->_nodeindices;
#endif
_cntml = _ml->_nodecount;
_thread = _ml->_thread;
for (_iml = 0; _iml < _cntml; ++_iml) {
_p = _ml->_data[_iml]; _ppvar = _ml->_pdata[_iml];
_tsav = -1e20;
#if CACHEVEC
if (use_cachevec) {
_v = VEC_V(_ni[_iml]);
}else
#endif
{
_nd = _ml->_nodelist[_iml];
_v = NODEV(_nd);
}
v = _v;
initmodel(_p, _ppvar, _thread, _nt);
}}
static double _nrn_current(double* _p, Datum* _ppvar, Datum* _thread, _NrnThread* _nt, double _v){double _current=0.;v=_v;{ {
mggate = 1.0 / ( 1.0 + exp ( 0.062 * - ( v ) ) * ( mg / 3.57 ) ) ;
g_AMPA = gmax * ( B_AMPA - A_AMPA ) ;
g_NMDA = gmax * ( B_NMDA - A_NMDA ) * mggate ;
g = g_AMPA + g_NMDA ;
i_AMPA = g_AMPA * ( v - e ) ;
i_NMDA = g_NMDA * ( v - e ) ;
i = i_AMPA + i_NMDA ;
}
_current += i;
} return _current;
}
static void nrn_cur(_NrnThread* _nt, _Memb_list* _ml, int _type) {
double* _p; Datum* _ppvar; Datum* _thread;
Node *_nd; int* _ni; double _rhs, _v; int _iml, _cntml;
#if CACHEVEC
_ni = _ml->_nodeindices;
#endif
_cntml = _ml->_nodecount;
_thread = _ml->_thread;
for (_iml = 0; _iml < _cntml; ++_iml) {
_p = _ml->_data[_iml]; _ppvar = _ml->_pdata[_iml];
#if CACHEVEC
if (use_cachevec) {
_v = VEC_V(_ni[_iml]);
}else
#endif
{
_nd = _ml->_nodelist[_iml];
_v = NODEV(_nd);
}
_g = _nrn_current(_p, _ppvar, _thread, _nt, _v + .001);
{ _rhs = _nrn_current(_p, _ppvar, _thread, _nt, _v);
}
_g = (_g - _rhs)/.001;
_g *= 1.e2/(_nd_area);
_rhs *= 1.e2/(_nd_area);
#if CACHEVEC
if (use_cachevec) {
VEC_RHS(_ni[_iml]) -= _rhs;
}else
#endif
{
NODERHS(_nd) -= _rhs;
}
}}
static void nrn_jacob(_NrnThread* _nt, _Memb_list* _ml, int _type) {
double* _p; Datum* _ppvar; Datum* _thread;
Node *_nd; int* _ni; int _iml, _cntml;
#if CACHEVEC
_ni = _ml->_nodeindices;
#endif
_cntml = _ml->_nodecount;
_thread = _ml->_thread;
for (_iml = 0; _iml < _cntml; ++_iml) {
_p = _ml->_data[_iml];
#if CACHEVEC
if (use_cachevec) {
VEC_D(_ni[_iml]) += _g;
}else
#endif
{
_nd = _ml->_nodelist[_iml];
NODED(_nd) += _g;
}
}}
static void nrn_state(_NrnThread* _nt, _Memb_list* _ml, int _type) {
double _break, _save;
double* _p; Datum* _ppvar; Datum* _thread;
Node *_nd; double _v; int* _ni; int _iml, _cntml;
#if CACHEVEC
_ni = _ml->_nodeindices;
#endif
_cntml = _ml->_nodecount;
_thread = _ml->_thread;
for (_iml = 0; _iml < _cntml; ++_iml) {
_p = _ml->_data[_iml]; _ppvar = _ml->_pdata[_iml];
_nd = _ml->_nodelist[_iml];
#if CACHEVEC
if (use_cachevec) {
_v = VEC_V(_ni[_iml]);
}else
#endif
{
_nd = _ml->_nodelist[_iml];
_v = NODEV(_nd);
}
_break = t + .5*dt; _save = t;
v=_v;
{
{ {
for (; t < _break; t += dt) {
{ state(_p, _ppvar, _thread, _nt); }
}}
t = _save;
}}}
}
static void terminal(){}
static void _initlists(){
double _x; double* _p = &_x;
int _i; static int _first = 1;
if (!_first) return;
_first = 0;
}
#if defined(__cplusplus)
} /* extern "C" */
#endif