2019-10-21 22:37:57 +00:00
// Hyperbolic Rogue -- raycaster
// Copyright (C) 2011-2019 Zeno Rogue, see 'hyper.cpp' for details
/** \file raycaster.cpp
* \ brief A raycaster to draw walls .
*/
# include "hyper.h"
namespace hr {
2020-03-27 20:49:20 +00:00
/** \brief raycaster */
2019-10-25 12:50:48 +00:00
EX namespace ray {
2020-03-21 09:49:07 +00:00
# if CAP_RAY
2019-11-03 13:19:11 +00:00
2019-10-25 12:50:48 +00:00
/** texture IDs */
2021-09-30 14:13:47 +00:00
GLuint txConnections = 0 , txWallcolor = 0 , txTextureMap = 0 , txVolumetric = 0 , txM = 0 , txWall = 0 , txPortalConnections = 0 ;
2019-10-21 22:37:57 +00:00
2019-10-25 12:50:48 +00:00
EX bool in_use ;
EX bool comparison_mode ;
/** 0 - never use, 2 - always use, 1 = smart selection */
EX int want_use = 1 ;
2021-02-18 14:47:45 +00:00
/** generate the map for raycasting just once */
EX bool fixed_map = false ;
2020-09-21 10:04:10 +00:00
EX ld exp_start = 1 ;
EX ld exp_decay_exp = 4 ;
EX ld exp_decay_poly = 10 ;
2019-10-25 13:05:17 +00:00
2020-10-18 12:38:21 +00:00
# ifdef GLES_ONLY
2020-10-15 14:37:43 +00:00
const int gms_limit = 16 ; /* enough for Bringris -- need to do better */
# else
2020-05-27 23:50:53 +00:00
const int gms_limit = 110 ;
2020-10-15 14:37:43 +00:00
# endif
2020-05-16 08:55:38 +00:00
2020-11-14 12:53:19 +00:00
EX int gms_array_size = 16 ;
2020-04-11 22:40:15 +00:00
EX ld maxstep_sol = .05 ;
2019-10-25 22:43:15 +00:00
EX ld maxstep_nil = .1 ;
2021-02-07 19:39:39 +00:00
EX ld maxstep_pro = .5 ;
2021-10-09 09:09:45 +00:00
EX ld maxstep_intra = .05 ;
2019-10-25 22:43:15 +00:00
EX ld minstep = .001 ;
2019-10-27 00:23:20 +00:00
EX ld reflect_val = 0 ;
2019-11-08 13:59:19 +00:00
static const int NO_LIMIT = 999999 ;
EX ld hard_limit = NO_LIMIT ;
2021-10-09 09:09:45 +00:00
EX int max_iter_intra = 600 ;
2020-10-18 12:48:35 +00:00
EX int max_iter_sol = 600 ;
EX int max_iter_iso = 60 ;
2021-02-07 19:39:39 +00:00
EX int max_iter_eyes = 200 ;
2019-10-25 13:18:31 +00:00
2019-10-26 16:25:14 +00:00
EX int max_cells = 2048 ;
2019-10-25 22:12:30 +00:00
EX bool rays_generate = true ;
2019-11-08 13:58:46 +00:00
EX ld & exp_decay_current ( ) {
2021-09-16 19:30:26 +00:00
if ( intra : : in ) return exp_decay_exp ;
2020-05-15 09:46:26 +00:00
if ( fake : : in ( ) ) return * FPIU ( & exp_decay_current ( ) ) ;
2020-05-09 08:41:15 +00:00
return ( sn : : in ( ) | | hyperbolic | | sl2 ) ? exp_decay_exp : exp_decay_poly ;
2019-10-25 21:24:31 +00:00
}
2019-11-08 13:58:46 +00:00
EX int & max_iter_current ( ) {
2021-10-09 09:09:45 +00:00
if ( intra : : in ) return max_iter_intra ;
2020-05-09 08:41:15 +00:00
if ( nonisotropic | | stretch : : in ( ) ) return max_iter_sol ;
2021-02-07 19:39:39 +00:00
else if ( is_eyes ( ) ) return max_iter_eyes ;
2019-10-25 21:24:31 +00:00
else return max_iter_iso ;
}
2021-02-07 19:39:39 +00:00
EX bool is_eyes ( ) {
# if CAP_VR
return vrhr : : active ( ) & & vrhr : : eyes = = vrhr : : eEyes : : equidistant ;
# else
return false ;
# endif
}
EX bool is_stepbased ( ) {
return nonisotropic | | stretch : : in ( ) | | is_eyes ( ) ;
}
2021-10-07 19:44:29 +00:00
EX bool horos ( ) {
return ( hyperbolic | | in_h2xe ( ) ) & & bt : : in ( ) ;
}
2019-10-25 22:43:15 +00:00
ld & maxstep_current ( ) {
2021-10-09 09:09:45 +00:00
if ( intra : : in ) return maxstep_intra ;
2020-05-09 08:41:15 +00:00
if ( sn : : in ( ) | | stretch : : in ( ) ) return maxstep_sol ;
2021-02-07 21:52:37 +00:00
# if CAP_VR
2021-02-07 19:39:39 +00:00
if ( vrhr : : active ( ) & & vrhr : : eyes = = vrhr : : eEyes : : equidistant )
return maxstep_pro ;
2021-02-07 21:52:37 +00:00
# endif
2021-02-07 19:39:39 +00:00
return maxstep_nil ;
2019-10-25 22:43:15 +00:00
}
2019-10-21 22:37:57 +00:00
# define IN_ODS 0
2019-10-26 15:03:10 +00:00
eGeometry last_geometry ;
2021-09-16 18:50:39 +00:00
vector < pair < int , cell * > > used_sample_list ( ) {
2021-09-16 19:30:26 +00:00
if ( intra : : in ) return intra : : full_sample_list ;
2021-09-16 18:50:39 +00:00
return hybrid : : gen_sample_list ( ) ;
}
2020-11-14 12:53:19 +00:00
bool need_many_cell_types ( ) {
2021-09-16 18:50:39 +00:00
return isize ( used_sample_list ( ) ) > 2 ;
2020-11-14 12:53:19 +00:00
}
2019-10-25 12:50:48 +00:00
/** is the raycaster available? */
EX bool available ( ) {
2020-11-19 17:20:06 +00:00
# if CAP_VR
2020-12-26 20:13:32 +00:00
/* would need a completely different implementation */
2021-01-31 14:27:58 +00:00
if ( vrhr : : active ( ) & & vrhr : : eyes = = vrhr : : eEyes : : equidistant ) {
if ( reflect_val ) return false ;
2021-02-06 12:03:09 +00:00
if ( sol | | stretch : : in ( ) | | sl2 ) return false ;
2021-01-31 14:27:58 +00:00
}
2020-11-19 17:20:06 +00:00
# endif
2019-11-09 10:18:52 +00:00
if ( noGUI ) return false ;
if ( ! vid . usingGL ) return false ;
2020-06-03 13:11:20 +00:00
if ( GDIM = = 2 ) return false ;
if ( WDIM = = 2 & & ( kite : : in ( ) | | bt : : in ( ) ) ) return false ;
2021-01-31 14:27:58 +00:00
# ifdef GLES_ONLY
2020-11-14 12:53:19 +00:00
if ( need_many_cell_types ( ) ) return false ;
2022-12-08 18:38:06 +00:00
if ( ! euclid & & ! gproduct & & ! nil ) return false ;
2020-11-14 12:53:19 +00:00
# endif
2019-12-14 11:12:24 +00:00
if ( hyperbolic & & pmodel = = mdPerspective & & ! kite : : in ( ) )
2019-10-25 12:50:48 +00:00
return true ;
2020-05-08 19:18:47 +00:00
if ( sphere & & pmodel = = mdPerspective & & ! rotspace )
return true ;
2019-11-23 19:39:38 +00:00
if ( nil & & S7 = = 8 )
return false ;
2020-05-09 08:41:15 +00:00
if ( ( sn : : in ( ) | | nil | | sl2 ) & & pmodel = = mdGeodesic )
2019-10-25 12:50:48 +00:00
return true ;
2019-12-14 11:05:01 +00:00
if ( euclid & & pmodel = = mdPerspective & & ! bt : : in ( ) )
2019-10-25 12:50:48 +00:00
return true ;
2022-12-08 18:38:06 +00:00
if ( gproduct )
2020-05-08 19:18:47 +00:00
return true ;
2020-05-09 08:41:15 +00:00
if ( pmodel = = mdPerspective & & stretch : : in ( ) )
2019-10-28 16:33:17 +00:00
return true ;
2019-10-25 12:50:48 +00:00
return false ;
}
/** do we want to use the raycaster? */
EX bool requested ( ) {
2020-03-29 15:35:42 +00:00
if ( cgflags & qRAYONLY ) return true ;
2019-10-25 12:50:48 +00:00
if ( ! want_use ) return false ;
2021-10-09 22:39:04 +00:00
if ( intra : : in ) return true ;
2020-07-28 14:45:04 +00:00
if ( stretch : : in ( ) & & sphere ) return true ;
2019-11-03 13:20:24 +00:00
# if CAP_TEXTURE
2019-11-03 10:00:47 +00:00
if ( texture : : config . tstate = = texture : : tsActive ) return false ;
2019-11-03 13:20:24 +00:00
# endif
2019-10-25 12:50:48 +00:00
if ( ! available ( ) ) return false ;
if ( want_use = = 2 ) return true ;
2020-07-27 16:49:27 +00:00
if ( rotspace ) return false ; // not very good
2022-03-27 17:14:57 +00:00
if ( WDIM = = 2 ) return false ; // not very good
2020-05-15 09:46:26 +00:00
return racing : : on | | quotient | | fake : : in ( ) ;
2019-10-25 12:50:48 +00:00
}
2020-03-29 15:35:42 +00:00
# if HDR
2019-10-21 22:37:57 +00:00
struct raycaster : glhr : : GLprogram {
2020-12-26 20:12:34 +00:00
GLint uStart , uStartid , uM , uLength , uIPD ;
2019-10-22 10:07:38 +00:00
GLint uWallstart , uWallX , uWallY ;
2020-07-07 20:16:38 +00:00
GLint tConnections , tWallcolor , tTextureMap , tVolumetric ;
2019-11-09 11:12:16 +00:00
GLint uBinaryWidth , uPLevel , uLP , uStraighten , uReflectX , uReflectY ;
2019-10-25 13:05:17 +00:00
GLint uLinearSightRange , uExpStart , uExpDecay ;
2019-11-02 09:40:22 +00:00
GLint uBLevel ;
2020-05-08 19:14:31 +00:00
GLint uWallOffset , uSides ;
2020-08-08 14:08:51 +00:00
GLint uITOA , uATOI ;
2020-08-22 22:10:59 +00:00
GLint uToOrig , uFromOrig ;
2020-12-26 20:12:34 +00:00
GLint uProjection ;
2021-01-31 14:27:58 +00:00
GLint uEyeShift , uAbsUnit ;
2021-07-09 18:12:35 +00:00
GLint uMirrorShift ;
2019-10-21 22:37:57 +00:00
2021-05-17 12:21:14 +00:00
GLint tM , uInvLengthM ;
GLint tWall , uInvLengthWall ;
2021-09-30 14:13:47 +00:00
GLint tPortalConnections ;
2020-03-29 15:35:42 +00:00
raycaster ( string vsh , string fsh ) ;
} ;
# endif
2021-06-01 16:03:18 +00:00
# ifdef GLES_ONLY
bool m_via_texture = false ;
bool wall_via_texture = false ;
static const bool can_via_texture = false ; /* textures are not precise enough ): */
# else
2021-05-17 12:21:14 +00:00
bool m_via_texture = true ;
bool wall_via_texture = true ;
2021-06-01 16:03:18 +00:00
static const bool can_via_texture = true ;
# endif
2021-05-17 12:21:14 +00:00
2020-03-29 15:35:42 +00:00
raycaster : : raycaster ( string vsh , string fsh ) : GLprogram ( vsh , fsh ) {
2021-06-09 02:44:04 +00:00
/* need to set shader_flags to 0 so that attributes are not enabled */
shader_flags = 0 ;
2019-10-21 22:37:57 +00:00
uStart = glGetUniformLocation ( _program , " uStart " ) ;
uStartid = glGetUniformLocation ( _program , " uStartid " ) ;
uM = glGetUniformLocation ( _program , " uM " ) ;
uLength = glGetUniformLocation ( _program , " uLength " ) ;
2020-12-26 20:12:34 +00:00
uProjection = glGetUniformLocation ( _program , " uProjection " ) ;
2019-10-21 22:37:57 +00:00
uIPD = glGetUniformLocation ( _program , " uIPD " ) ;
2019-10-22 10:07:38 +00:00
uWallstart = glGetUniformLocation ( _program , " uWallstart " ) ;
uWallX = glGetUniformLocation ( _program , " uWallX " ) ;
uWallY = glGetUniformLocation ( _program , " uWallY " ) ;
2019-10-22 20:42:48 +00:00
uBinaryWidth = glGetUniformLocation ( _program , " uBinaryWidth " ) ;
2019-11-08 14:01:03 +00:00
uStraighten = glGetUniformLocation ( _program , " uStraighten " ) ;
2019-10-28 16:33:17 +00:00
uPLevel = glGetUniformLocation ( _program , " uPLevel " ) ;
uLP = glGetUniformLocation ( _program , " uLP " ) ;
2019-11-09 11:12:16 +00:00
uReflectX = glGetUniformLocation ( _program , " uReflectX " ) ;
uReflectY = glGetUniformLocation ( _program , " uReflectY " ) ;
2021-07-09 18:12:35 +00:00
uMirrorShift = glGetUniformLocation ( _program , " uMirrorShift " ) ;
2019-10-25 13:05:17 +00:00
uLinearSightRange = glGetUniformLocation ( _program , " uLinearSightRange " ) ;
uExpDecay = glGetUniformLocation ( _program , " uExpDecay " ) ;
uExpStart = glGetUniformLocation ( _program , " uExpStart " ) ;
2019-11-02 09:40:22 +00:00
uBLevel = glGetUniformLocation ( _program , " uBLevel " ) ;
2019-10-21 22:37:57 +00:00
tConnections = glGetUniformLocation ( _program , " tConnections " ) ;
tWallcolor = glGetUniformLocation ( _program , " tWallcolor " ) ;
2019-10-26 08:32:44 +00:00
tTextureMap = glGetUniformLocation ( _program , " tTextureMap " ) ;
2020-07-07 20:16:38 +00:00
tVolumetric = glGetUniformLocation ( _program , " tVolumetric " ) ;
2021-05-17 12:21:14 +00:00
2021-09-30 14:13:47 +00:00
tPortalConnections = glGetUniformLocation ( _program , " tPortalConnections " ) ;
2021-05-17 12:21:14 +00:00
tM = glGetUniformLocation ( _program , " tM " ) ;
uInvLengthM = glGetUniformLocation ( _program , " uInvLengthM " ) ;
tWall = glGetUniformLocation ( _program , " tWall " ) ;
uInvLengthWall = glGetUniformLocation ( _program , " uInvLengthWall " ) ;
2020-02-07 18:13:18 +00:00
2020-05-08 19:14:31 +00:00
uWallOffset = glGetUniformLocation ( _program , " uWallOffset " ) ;
uSides = glGetUniformLocation ( _program , " uSides " ) ;
2020-08-08 14:08:51 +00:00
uITOA = glGetUniformLocation ( _program , " uITOA " ) ;
uATOI = glGetUniformLocation ( _program , " uATOI " ) ;
2020-08-22 22:10:59 +00:00
uToOrig = glGetUniformLocation ( _program , " uToOrig " ) ;
uFromOrig = glGetUniformLocation ( _program , " uFromOrig " ) ;
2021-01-31 14:27:58 +00:00
uEyeShift = glGetUniformLocation ( _program , " uEyeShift " ) ;
uAbsUnit = glGetUniformLocation ( _program , " uAbsUnit " ) ;
2019-10-21 22:37:57 +00:00
}
shared_ptr < raycaster > our_raycaster ;
2020-01-15 17:39:35 +00:00
# ifdef GLES_ONLY
void add ( string & tgt , string type , string name , int min_index , int max_index ) {
2020-10-15 14:37:43 +00:00
if ( min_index > = max_index ) ;
else
2020-01-15 17:39:35 +00:00
if ( min_index + 1 = = max_index )
tgt + = " { return " + name + " [ " + its ( min_index ) + " ]; } " ;
else {
int mid = ( min_index + max_index ) / 2 ;
tgt + = " { if(i< " + its ( mid ) + " ) " ;
add ( tgt , type , name , min_index , mid ) ;
tgt + = " else " ;
add ( tgt , type , name , mid , max_index ) ;
tgt + = " } " ;
}
}
string build_getter ( string type , string name , int index ) {
string s = type + " get_ " + name + " (int i) \n " ;
add ( s , type , name , 0 , index ) ;
return s + " \n " ;
}
# define GET(array, index) "get_" array "(" index ")"
# else
# define GET(array, index) array "[" index "]"
# endif
2020-04-11 18:47:14 +00:00
EX hookset < void ( string & , string & ) > hooks_rayshader ;
EX hookset < bool ( shared_ptr < raycaster > ) > hooks_rayset ;
2020-03-29 15:35:42 +00:00
2021-02-07 17:30:20 +00:00
tuple <
# if CAP_VR
int , vrhr : : eEyes ,
# endif
2021-04-07 00:33:08 +00:00
string
2021-02-07 17:30:20 +00:00
> raycaster_state ( ) {
return make_tuple (
# if CAP_VR
vrhr : : state ,
vrhr : : eyes ,
# endif
2021-10-09 22:39:04 +00:00
intra : : in ? " INTRA " : cgi_string ( )
2021-02-07 17:30:20 +00:00
) ;
}
decltype ( raycaster_state ( ) ) saved_state ;
2021-07-07 21:33:34 +00:00
ld ray_scale = 8 ;
int max_wall_offset = 512 ;
int max_celltype = 64 ;
2021-09-16 18:50:39 +00:00
struct raygen {
string fsh , vsh , fmain ;
2021-05-17 12:21:14 +00:00
2021-10-03 08:55:07 +00:00
void add_if ( const string & seek , const string & function ) ;
2021-09-16 18:50:39 +00:00
int deg , irays ;
bool asonov ;
bool use_reflect ;
bool many_cell_types ;
bool eyes ;
string getM ( string s ) {
2021-05-17 12:21:14 +00:00
if ( m_via_texture )
return " getM( " + s + " ) " ;
else
return " uM[ " + s + " ] " ;
} ;
2021-09-16 18:50:39 +00:00
string getWall ( string s , int coord ) {
2021-05-17 12:21:14 +00:00
if ( wall_via_texture )
return " getWall( " + s + " , " + its ( coord ) + " ) " ;
else
return " uWall " + string ( coord ? " Y " : " X " ) + " [ " + s + " ] " ;
} ;
2021-09-16 18:50:39 +00:00
string getWallstart ( string s ) {
2021-05-17 12:21:14 +00:00
if ( wall_via_texture )
return " getWallstart( " + s + " ) " ;
else
return " uWallstart[ " + s + " ] " ;
} ;
2020-05-08 19:14:31 +00:00
2021-10-03 08:55:07 +00:00
void compute_which_and_dist ( int flat1 , int flat2 ) ;
void apply_reflect ( int flat1 , int flat2 ) ;
2021-09-16 18:50:39 +00:00
void move_forward ( ) ;
2021-09-16 19:30:26 +00:00
void emit_intra_portal ( int gid1 , int gid2 ) ;
void emit_iterate ( int gid1 ) ;
2021-10-09 09:11:31 +00:00
void emit_raystarter ( ) ;
2021-09-16 18:50:39 +00:00
void create ( ) ;
2021-10-03 08:55:07 +00:00
2021-10-07 19:44:29 +00:00
string f_xpush ( ) { return hyperbolic ? " xpush_h3 " : " xpush_s3 " ; }
string f_len ( ) { return hyperbolic ? " len_h3 " : ( sphere & & rotspace ) ? " len_sr " : sl2 ? " len_sl2 " : sphere ? " len_s3 " : " len_x " ; }
string f_len_prod ( ) { return in_h2xe ( ) ? " len_h2 " : in_s2xe ( ) ? " len_s2 " : " len_e2 " ; }
2021-10-03 08:55:07 +00:00
void add_functions ( ) ;
2021-09-16 18:50:39 +00:00
} ;
raygen our_raygen ;
2021-10-03 08:55:07 +00:00
void raygen : : add_if ( const string & seek , const string & function ) {
if ( fsh . find ( seek ) ! = string : : npos | | fmain . find ( seek ) ! = string : : npos )
fsh = function + fsh ;
}
void raygen : : compute_which_and_dist ( int flat1 , int flat2 ) {
2021-09-16 18:50:39 +00:00
using glhr : : to_glsl ;
2021-10-03 16:03:14 +00:00
if ( ! is_stepbased ( ) ) {
2021-09-16 18:50:39 +00:00
fmain + =
" if(which == -1) { \n " ;
2021-10-07 20:26:41 +00:00
if ( in_h2xe ( ) & & hybrid : : underlying = = gBinary4 )
fmain + = " for(int i=2; i<=4; i+=2) { " ;
else if ( in_h2xe ( ) & & hybrid : : underlying = = gTernary )
fmain + = " for(int i=3; i<=5; i+=2) { " ;
else if ( in_h2xe ( ) & & hybrid : : underlying = = gBinaryTiling )
fmain + = " for(int i=0; i<=4; i++) if(i == 0 || i == 4) { " ;
2021-10-07 19:44:29 +00:00
else
2022-12-08 18:38:06 +00:00
fmain + = " for(int i= " + its ( flat1 ) + " ; i< " + ( gproduct ? " sides-2 " : ( ( WDIM = = 2 | | is_subcube_based ( variation ) | | intra : : in ) & & ! bt : : in ( ) ) ? " sides " : its ( flat2 ) ) + " ; i++) { \n " ;
2021-09-16 18:50:39 +00:00
2021-09-16 20:11:09 +00:00
fmain + = " mediump mat4 m = " + getM ( " walloffset+i " ) + " ; \n " ;
2021-09-16 18:50:39 +00:00
if ( in_h2xe ( ) ) fmain + =
2021-09-16 20:11:09 +00:00
" mediump float v = ((position - m * position)[2] / (m * tangent - tangent)[2]); \n "
2021-09-16 18:50:39 +00:00
" if(v > 1. || v < -1.) continue; \n "
" mediump float d = atanh(v); \n "
" mediump vec4 next_tangent = position * sinh(d) + tangent * cosh(d); \n "
2021-09-16 20:11:09 +00:00
" if(next_tangent[2] < (m * next_tangent)[2]) continue; \n "
2021-09-16 18:50:39 +00:00
" d /= xspeed; \n " ;
else if ( in_s2xe ( ) ) fmain + =
2021-09-16 20:11:09 +00:00
" mediump float v = ((position - m * position)[2] / (m * tangent - tangent)[2]); \n "
2021-09-16 18:50:39 +00:00
" mediump float d = atan(v); \n "
" mediump vec4 next_tangent = tangent * cos(d) - position * sin(d); \n "
2021-09-16 20:11:09 +00:00
" if(next_tangent[2] > (m * next_tangent)[2]) continue; \n "
2021-09-16 18:50:39 +00:00
" d /= xspeed; \n " ;
else if ( in_e2xe ( ) ) fmain + =
2021-09-16 20:11:09 +00:00
" mediump float deno = dot(position, tangent) - dot(m*position, m*tangent); \n "
2021-09-16 18:50:39 +00:00
" if(deno < 1e-6 && deno > -1e-6) continue; \n "
2021-09-16 20:11:09 +00:00
" mediump float d = (dot(m*position, m*position) - dot(position, position)) / 2. / deno; \n "
2021-09-16 18:50:39 +00:00
" if(d < 0.) continue; \n "
" mediump vec4 next_position = position + d * tangent; \n "
2021-09-16 20:11:09 +00:00
" if(dot(next_position, tangent) < dot(m*next_position, m*tangent)) continue; \n "
2021-09-16 18:50:39 +00:00
" d /= xspeed; \n " ;
2022-12-15 20:17:08 +00:00
else if ( geom3 : : sph_in_hyp ( ) ) fmain + =
" mediump float v = ((zpush_h3(-1.) * (position - m * position))[3] / (zpush_h3(-1.) * (m * tangent - tangent))[3]); \n "
" if(v > 1. || v < -1.) continue; \n "
" mediump float d = atanh(v); \n "
" mediump vec4 next_tangent = position * sinh(d) + tangent * cosh(d); \n "
" if((zpush_h3(-1.) * next_tangent)[3] < (zpush_h3(-1.) * (m * next_tangent))[3]) continue; \n " ;
2021-09-16 18:50:39 +00:00
else if ( hyperbolic ) fmain + =
2021-09-16 20:11:09 +00:00
" mediump float v = ((position - m * position)[3] / (m * tangent - tangent)[3]); \n "
2021-09-16 18:50:39 +00:00
" if(v > 1. || v < -1.) continue; \n "
" mediump float d = atanh(v); \n "
" mediump vec4 next_tangent = position * sinh(d) + tangent * cosh(d); \n "
2021-09-16 20:11:09 +00:00
" if(next_tangent[3] < (m * next_tangent)[3]) continue; \n " ;
2021-09-16 18:50:39 +00:00
else if ( sphere ) fmain + =
2021-09-16 20:11:09 +00:00
" mediump float v = ((position - m * position)[3] / (m * tangent - tangent)[3]); \n "
2021-09-16 18:50:39 +00:00
" mediump float d = atan(v); \n "
" mediump vec4 next_tangent = -position * sin(d) + tangent * cos(d); \n "
2021-09-16 20:11:09 +00:00
" if(next_tangent[3] > (m * next_tangent)[3]) continue; \n " ;
2022-12-15 20:17:08 +00:00
else if ( geom3 : : sph_in_euc ( ) ) fmain + =
" vec4 tctr = vec4(0, 0, 1, 0); \n "
" mediump float deno = dot(position-tctr, tangent) - dot(m*position-tctr, m*tangent); \n "
" if(deno < 1e-6 && deno > -1e-6) continue; \n "
" mediump float d = (dot(m*position-tctr, m*position-tctr) - dot(position-tctr, position-tctr)) / 2. / deno; \n "
" if(d < 0.) continue; \n "
" mediump vec4 next_position = position + d * tangent; \n "
" if(dot(next_position - tctr, tangent) < dot(m*next_position - tctr, m*tangent)) continue; \n " ;
2021-09-16 18:50:39 +00:00
else fmain + =
2021-09-16 20:11:09 +00:00
" mediump float deno = dot(position, tangent) - dot(m*position, m*tangent); \n "
2021-09-16 18:50:39 +00:00
" if(deno < 1e-6 && deno > -1e-6) continue; \n "
2021-09-16 20:11:09 +00:00
" mediump float d = (dot(m*position, m*position) - dot(position, position)) / 2. / deno; \n "
2021-09-16 18:50:39 +00:00
" if(d < 0.) continue; \n "
" mediump vec4 next_position = position + d * tangent; \n "
2021-09-16 20:11:09 +00:00
" if(dot(next_position, tangent) < dot(m*next_position, m*tangent)) continue; \n " ;
2021-09-16 18:50:39 +00:00
fmain + =
" if(d < dist) { dist = d; which = i; } \n "
" } \n " ;
if ( hyperbolic & & reg3 : : ultra_mirror_in ( ) ) {
fmain + = " for(int i= " + its ( S7 * 2 ) + " ; i< " + its ( S7 * 2 + isize ( cgi . heptshape - > vertices_only ) ) + " ; i++) { \n " ;
fmain + = " mat4 uMi = " + getM ( " i " ) + " ; " ;
fmain + =
" mediump float v = ((position - uMi * position)[3] / (uMi * tangent - tangent)[3]); \n "
" if(v > 1. || v < -1.) continue; \n "
" mediump float d = atanh(v); \n "
" mediump vec4 next_tangent = position * sinh(d) + tangent * cosh(d); \n "
" if(next_tangent[3] < (uMi * next_tangent)[3]) continue; \n "
" if(d < dist) { dist = d; which = i; } \n "
" } \n " ;
}
// 20: get to horosphere +uBLevel (take smaller root)
// 21: get to horosphere -uBLevel (take larger root)
2021-10-07 19:44:29 +00:00
if ( horos ( ) ) {
string push = hyperbolic ? " xpush_h3 " : " xpush_h2 " ;
string w = hyperbolic ? " w " : " z " ;
2021-09-16 18:50:39 +00:00
fmain + =
" for(int i=20; i<22; i++) { \n "
" mediump float sgn = i == 20 ? -1. : 1.; \n "
2021-10-07 19:44:29 +00:00
" mediump vec4 zpos = " + push + " (uBLevel*sgn) * position; \n "
" mediump vec4 ztan = " + push + " (uBLevel*sgn) * tangent; \n "
" mediump float Mp = zpos. " + w + " - zpos.x; \n "
" mediump float Mt = ztan. " + w + " - ztan.x; \n "
2021-09-16 18:50:39 +00:00
" mediump float a = (Mp*Mp-Mt*Mt); \n "
" mediump float b = Mp/a; \n "
" mediump float c = (1.+Mt*Mt) / a; \n "
" if(b*b < c) continue; \n "
" if(sgn < 0. && Mt > 0.) continue; \n "
" mediump float zsgn = (Mt > 0. ? -sgn : sgn); \n "
" mediump float u = sqrt(b*b-c)*zsgn + b; \n "
" mediump float v = -(Mp*u-1.) / Mt; \n "
2022-02-26 08:50:45 +00:00
" if(a < 1e-5) v = (1.-Mp*Mp) / (2. * Mt); \n "
2021-10-07 19:44:29 +00:00
" mediump float d = asinh(v); \n " ;
2022-12-08 18:38:06 +00:00
if ( gproduct ) fmain + = " d /= xspeed; \n " ;
2021-10-07 19:44:29 +00:00
fmain + =
" if(d < 0. && abs(log(position. " + w + " *position. " + w + " -position.x*position.x)) < uBLevel) continue; \n "
2021-09-16 18:50:39 +00:00
" if(d < dist) { dist = d; which = i; } \n "
" } \n " ;
}
2022-12-08 18:38:06 +00:00
if ( gproduct ) fmain + =
2021-09-16 18:50:39 +00:00
" if(zspeed > 0.) { mediump float d = (uPLevel - zpos) / zspeed; if(d < dist) { dist = d; which = sides-1; }} \n "
" if(zspeed < 0.) { mediump float d = (-uPLevel - zpos) / zspeed; if(d < dist) { dist = d; which = sides-2; }} \n " ;
fmain + = " } \n " ;
fmain + =
" if(dist < 0.) { dist = 0.; } \n " ;
fmain + =
" if(which == -1 && dist == 0.) return; \n " ;
}
}
void raygen : : move_forward ( ) {
using glhr : : to_glsl ;
2021-10-03 16:03:14 +00:00
if ( in_h2xe ( ) & & ! is_stepbased ( ) ) fmain + =
2021-09-16 18:50:39 +00:00
" mediump float ch = cosh(dist*xspeed); mediump float sh = sinh(dist*xspeed); \n "
" mediump vec4 v = position * ch + tangent * sh; \n "
" tangent = tangent * ch + position * sh; \n "
" position = v; \n "
" zpos += dist * zspeed; \n " ;
2021-10-03 16:03:14 +00:00
else if ( in_s2xe ( ) & & ! is_stepbased ( ) ) fmain + =
2021-09-16 18:50:39 +00:00
" mediump float ch = cos(dist*xspeed); mediump float sh = sin(dist*xspeed); \n "
" mediump vec4 v = position * ch + tangent * sh; \n "
" tangent = tangent * ch - position * sh; \n "
" position = v; \n "
" zpos += dist * zspeed; \n " ;
2021-10-03 16:03:14 +00:00
else if ( in_e2xe ( ) & & ! is_stepbased ( ) ) fmain + =
2021-09-16 18:50:39 +00:00
" position = position + tangent * dist * xspeed; \n "
" zpos += dist * zspeed; \n " ;
2021-10-03 16:03:14 +00:00
else if ( hyperbolic & & ! is_stepbased ( ) ) fmain + =
2021-09-16 18:50:39 +00:00
" mediump float ch = cosh(dist); mediump float sh = sinh(dist); \n "
" mediump vec4 v = position * ch + tangent * sh; \n "
" tangent = tangent * ch + position * sh; \n "
" position = v; \n " ;
2021-10-03 16:03:14 +00:00
else if ( sphere & & ! is_stepbased ( ) ) fmain + =
2021-09-16 18:50:39 +00:00
" mediump float ch = cos(dist); mediump float sh = sin(dist); \n "
" mediump vec4 v = position * ch + tangent * sh; \n "
" tangent = tangent * ch - position * sh; \n "
" position = v; \n " ;
2021-10-03 16:03:14 +00:00
else if ( is_stepbased ( ) ) {
2021-09-16 18:50:39 +00:00
bool use_christoffel = true ;
if ( sol & & nih ) fsh + =
" mediump vec4 christoffel(mediump vec4 pos, mediump vec4 vel, mediump vec4 tra) { \n "
" return vec4(-(vel.z*tra.x + vel.x*tra.z)*log(2.), (vel.z*tra.y + vel.y * tra.z)*log(3.), vel.x*tra.x * exp(2.*log(2.)*pos.z)*log(2.) - vel.y * tra.y * exp(-2.*log(3.)*pos.z)*log(3.), 0.); \n "
" } \n " ;
else if ( nih ) fsh + =
" mediump vec4 christoffel(mediump vec4 pos, mediump vec4 vel, mediump vec4 tra) { \n "
" return vec4((vel.z*tra.x + vel.x*tra.z)*log(2.), (vel.z*tra.y + vel.y * tra.z)*log(3.), -vel.x*tra.x * exp(-2.*log(2.)*pos.z)*log(2.) - vel.y * tra.y * exp(-2.*log(3.)*pos.z)*log(3.), 0.); \n "
" } \n " ;
else if ( sol ) fsh + =
" mediump vec4 christoffel(mediump vec4 pos, mediump vec4 vel, mediump vec4 tra) { \n "
" return vec4(-vel.z*tra.x - vel.x*tra.z, vel.z*tra.y + vel.y * tra.z, vel.x*tra.x * exp(2.*pos.z) - vel.y * tra.y * exp(-2.*pos.z), 0.); \n "
" } \n " ;
else if ( nil ) {
fsh + =
" mediump vec4 christoffel(mediump vec4 pos, mediump vec4 vel, mediump vec4 tra) { \n "
" mediump float x = pos.x; \n "
" return vec4(x*vel.y*tra.y - 0.5*dot(vel.yz,tra.zy), -.5*x*dot(vel.yx,tra.xy) + .5 * dot(vel.zx,tra.xz), -.5*(x*x-1.)*dot(vel.yx,tra.xy)+.5*x*dot(vel.zx,tra.xz), 0.); \n "
// " return vec4(0.,0.,0.,0.);\n"
" } \n " ;
use_christoffel = false ;
}
else if ( sl2 | | stretch : : in ( ) ) {
if ( sl2 ) {
fsh + = " mediump mat4 s_translate(vec4 h) { \n "
" return mat4(h.w,h.z,h.y,h.x,-h.z,h.w,-h.x,h.y,h.y,-h.x,h.w,-h.z,h.x,h.y,h.z,h.w); \n "
" } \n " ;
}
else {
fsh + = " mediump mat4 s_translate(vec4 h) { \n "
" return mat4(h.w,h.z,-h.y,-h.x,-h.z,h.w,h.x,-h.y,h.y,-h.x,h.w,-h.z,h.x,h.y,h.z,h.w); \n "
" } \n " ;
}
fsh + = " mediump mat4 s_itranslate(vec4 h) { \n "
" h.xyz = -h.xyz; return s_translate(h); \n "
" } \n " ;
if ( stretch : : mstretch ) {
fsh + = " mediump vec4 christoffel(mediump vec4 pos, mediump vec4 vel, mediump vec4 tra) { \n "
" vel = s_itranslate(toOrig * pos) * toOrig * vel; \n "
" tra = s_itranslate(toOrig * pos) * toOrig * tra; \n "
" return fromOrig * s_translate(toOrig * pos) * vec4( \n " ;
for ( int i = 0 ; i < 3 ; i + + ) {
auto & c = stretch : : ms_christoffel ;
fsh + = " 0. " ;
for ( int j = 0 ; j < 3 ; j + + )
for ( int k = 0 ; k < 3 ; k + + )
if ( c [ i ] [ j ] [ k ] )
fsh + = " + vel[ " + its ( j ) + " ]*tra[ " + its ( k ) + " ]* " + to_glsl ( c [ i ] [ j ] [ k ] ) ;
fsh + = " , \n " ;
}
fsh + = " 0); \n "
" } \n " ;
}
else
use_christoffel = false ;
}
else use_christoffel = false ;
if ( use_christoffel ) fsh + = " mediump vec4 get_acc(mediump vec4 pos, mediump vec4 vel) { \n "
" return christoffel(pos, vel, vel); \n "
" } \n " ;
if ( sn : : in ( ) & & ! asonov ) fsh + = " uniform mediump float uBinaryWidth; \n " ;
fmain + =
" dist = next < minstep ? 2.*next : next; \n " ;
if ( nil & & ! use_christoffel ) fsh + =
" mediump vec4 translate(mediump vec4 a, mediump vec4 b) { \n "
" return vec4(a[0] + b[0], a[1] + b[1], a[2] + b[2] + a[0] * b[1], b[3]); \n "
" } \n "
" mediump vec4 translatev(mediump vec4 a, mediump vec4 t) { \n "
" return vec4(t[0], t[1], t[2] + a[0] * t[1], 0.); \n "
" } \n "
" mediump vec4 itranslate(mediump vec4 a, mediump vec4 b) { \n "
" return vec4(-a[0] + b[0], -a[1] + b[1], -a[2] + b[2] - a[0] * (b[1]-a[1]), b[3]); \n "
" } \n "
" mediump vec4 itranslatev(mediump vec4 a, mediump vec4 t) { \n "
" return vec4(t[0], t[1], t[2] - a[0] * t[1], 0.); \n "
" } \n " ;
// if(nil) fmain += "tangent = translate(position, itranslate(position, tangent));\n";
if ( use_christoffel ) fmain + =
" mediump vec4 vel = tangent * dist; \n "
" mediump vec4 acc1 = get_acc(position, vel); \n "
" mediump vec4 acc2 = get_acc(position + vel / 2., vel + acc1/2.); \n "
" mediump vec4 acc3 = get_acc(position + vel / 2. + acc1/4., vel + acc2/2.); \n "
" mediump vec4 acc4 = get_acc(position + vel + acc2/2., vel + acc3/2.); \n "
" mediump vec4 nposition = position + vel + (acc1+acc2+acc3)/6.; \n " ;
if ( ( sl2 | | stretch : : in ( ) ) & & use_christoffel ) {
if ( sl2 ) fmain + =
" nposition = nposition / sqrt(dot(position.zw, position.zw) - dot(nposition.xy, nposition.xy)); \n " ;
else if ( stretch : : in ( ) ) fmain + =
" nposition = nposition / sqrt(dot(nposition, nposition)); \n " ;
}
if ( ( sl2 | | stretch : : in ( ) ) & & ! use_christoffel ) {
ld SV = stretch : : not_squared ( ) ;
ld mul = ( sphere ? 1 : - 1 ) - 1 / SV / SV ;
fmain + =
" vec4 vel = s_itranslate(position) * tangent * dist; \n "
" vec4 vel1 = vel; vel1.z *= " + to_glsl ( stretch : : not_squared ( ) ) + " ; \n "
" mediump float vlen = length(vel1.xyz); \n "
" if(vel.z<0.) vlen=-vlen; \n "
" float z_part = vel1.z/vlen; \n "
" float x_part = sqrt(1.-z_part*z_part); \n "
" const float SV = " + to_glsl ( SV ) + " ; \n "
" float rparam = x_part / z_part / SV; \n "
" float beta = atan2(vel.y,vel.x); \n "
" if(vlen<0.) beta += PI; \n "
" mediump vec4 nposition, ntangent; \n " ;
if ( sl2 ) fmain + =
" if(rparam > 1.) { \n "
" float cr = 1./sqrt(rparam*rparam-1.); \n "
" float sr = rparam*cr; \n "
" float z = cr * " + to_glsl ( mul ) + " ; \n "
" float a = vlen / length(vec2(sr, cr/SV)); \n "
" float k = -a; \n "
" float u = z*a; \n "
" float xy = sr * sinh(k); \n "
" float zw = cr * sinh(k); \n "
" nposition = vec4( "
" -xy*cos(u+beta), "
" -xy*sin(u+beta), "
" zw*cos(u)-cosh(k)*sin(u), "
" zw*sin(u)+cosh(k)*cos(u) "
" ); \n "
" ntangent = vec4( "
" -sr*cosh(k)*k*cos(u+beta) + u*xy*sin(u+beta), "
" -sr*cosh(k)*k*sin(u+beta) - u*xy*cos(u+beta), "
" k*cr*cosh(k)*cos(u)-zw*sin(u)*u-k*sinh(k)*sin(u)-u*cosh(k)*cos(u), "
" k*cr*cosh(k)*sin(u)+u*zw*cos(u)+k*sinh(k)*cos(u)-u*cosh(k)*sin(u) "
" ); \n "
" } \n "
" else { \n "
" float r = atanh(rparam); \n "
" float cr = cosh(r); \n "
" float sr = sinh(r); \n "
" float z = cr * " + to_glsl ( mul ) + " ; \n "
" float a = vlen / length(vec2(sr, cr/SV)); \n "
" float k = -a; \n "
" float u = z*a; \n "
" float xy = sr * sin(k); \n "
" float zw = cr * sin(k); \n "
" ntangent = vec4( "
" -sr*cos(k)*k*cos(u+beta) + u*xy*sin(u+beta), "
" -sr*cos(k)*k*sin(u+beta) - u*xy*cos(u+beta), "
" k*cr*cos(k)*cos(u)-zw*sin(u)*u+k*sin(k)*sin(u)-u*cos(k)*cos(u), "
" k*cr*cos(k)*sin(u)+zw*cos(u)*u-k*sin(k)*cos(u)-u*cos(k)*sin(u) "
" ); \n "
" nposition = vec4( "
" -xy * cos(u+beta), "
" -xy * sin(u+beta), "
" zw * cos(u) - cos(k) * sin(u), "
" zw * sin(u) + cos(k)*cos(u) "
" ); \n "
" } \n " ;
else fmain + =
" if(true) { \n "
" float r = atan(rparam); \n "
" float cr = cos(r); \n "
" float sr = sin(r); \n "
" float z = cr * " + to_glsl ( mul ) + " ; \n "
" float a = vlen / length(vec2(sr, cr/SV)); \n "
" float k = a; \n "
" float u = z*a; \n "
" float xy = sr * sin(k); \n "
" float zw = cr * sin(k); \n "
" ntangent = vec4( "
" sr*cos(k)*k*cos(u+beta) - u*xy*sin(u+beta), "
" sr*cos(k)*k*sin(u+beta) + u*xy*cos(u+beta), "
" k*cr*cos(k)*cos(u)-zw*sin(u)*u+k*sin(k)*sin(u)-u*cos(k)*cos(u), "
" k*cr*cos(k)*sin(u)+zw*cos(u)*u-k*sin(k)*cos(u)-u*cos(k)*sin(u) "
" ); \n "
" nposition = vec4( "
" xy * cos(u+beta), "
" xy * sin(u+beta), "
" zw * cos(u) - cos(k) * sin(u), "
" zw * sin(u) + cos(k)*cos(u) "
" ); \n "
" } \n " ;
fmain + =
" ntangent = ntangent / dist; \n "
" ntangent = s_translate(position) * ntangent; \n "
" nposition = s_translate(position) * nposition; \n " ;
}
if ( nil & & ! use_christoffel & & ! eyes ) {
fmain + =
" mediump vec4 xp, xt; \n "
" mediump vec4 back = itranslatev(position, tangent); \n "
" if(back.x == 0. && back.y == 0.) { \n "
" xp = vec4(0., 0., back.z*dist, 1.); \n "
" xt = back; \n "
" } \n "
" else if(abs(back.z) == 0.) { \n "
" xp = vec4(back.x*dist, back.y*dist, back.x*back.y*dist*dist/2., 1.); \n "
" xt = vec4(back.x, back.y, dist*back.x*back.y, 0.); \n "
" } \n "
" else if(abs(back.z) < 1e-1) { \n "
// we use the midpoint method here, because the formulas below cause glitches due to mediump float precision
" mediump vec4 acc = christoffel(vec4(0,0,0,1), back, back); \n "
" mediump vec4 pos2 = back * dist / 2.; \n "
" mediump vec4 tan2 = back + acc * dist / 2.; \n "
" mediump vec4 acc2 = christoffel(pos2, tan2, tan2); \n "
" xp = vec4(0,0,0,1) + back * dist + acc2 / 2. * dist * dist; \n "
" xt = back + acc * dist; \n "
" } \n "
" else { \n "
" mediump float alpha = atan2(back.y, back.x); \n "
" mediump float w = back.z * dist; \n "
" mediump float c = length(back.xy) / back.z; \n "
" xp = vec4(2.*c*sin(w/2.) * cos(w/2.+alpha), 2.*c*sin(w/2.)*sin(w/2.+alpha), w*(1.+(c*c/2.)*((1.-sin(w)/w)+(1.-cos(w))/w * sin(w+2.*alpha))), 1.); \n "
" xt = back.z * vec4( "
" c*cos(alpha+w), "
" c*sin(alpha+w), "
" 1. + c*c*2.*sin(w/2.)*sin(alpha+w)*cos(alpha+w/2.), "
" 0.); \n "
" } \n "
" mediump vec4 nposition = translate(position, xp); \n " ;
}
if ( asonov ) {
fsh + = " uniform mediump mat4 uStraighten; \n " ;
fmain + = " mediump vec4 sp = uStraighten * nposition; \n " ;
}
if ( eyes ) {
fmain + =
" mediump float t = go + dist; \n " ;
2022-12-08 18:38:06 +00:00
fmain + = gproduct ?
2021-09-16 18:50:39 +00:00
" mediump vec4 v = at1 * t; \n " :
" mediump vec4 v = at0 * t; \n " ;
fmain + =
" v[3] = 1.; \n "
" mediump vec4 azeq = uEyeShift * v; \n " ;
if ( nil ) fmain + =
" mediump float alpha = atan2(azeq.y, azeq.x); \n "
" mediump float w = azeq.z; \n "
" mediump float c = length(azeq.xy) / azeq.z; \n "
" mediump vec4 xp = vec4(2.*c*sin(w/2.) * cos(w/2.+alpha), 2.*c*sin(w/2.)*sin(w/2.+alpha), w*(1.+(c*c/2.)*((1.-sin(w)/w)+(1.-cos(w))/w * sin(w+2.*alpha))), 1.); \n "
" mediump vec4 orig_position = vw * vec4(0., 0., 0., 1.); \n "
" mediump vec4 nposition = translate(orig_position, xp); \n " ;
2022-12-08 18:38:06 +00:00
else if ( gproduct ) {
2021-09-16 18:50:39 +00:00
fmain + =
" mediump float alen_xy = length(azeq.xy); \n " ;
fmain + = " mediump float nzpos = zpos + azeq.z; \n " ;
if ( in_h2xe ( ) ) {
fmain + = " azeq.xy *= sinh(alen_xy) / alen_xy; \n " ;
fmain + = " azeq.z = cosh(alen_xy); \n " ;
}
else if ( in_s2xe ( ) ) {
fmain + = " azeq.xy *= sin (alen_xy) / alen_xy; \n " ;
fmain + = " azeq.z = cos(alen_xy); \n " ;
}
else {
/* euclid */
fmain + = " azeq.z = 1.; \n " ;
}
fmain + = " azeq.w = 0.; \n " ;
fmain + =
" mediump vec4 nposition = vw * azeq; \n " ;
}
else {
fmain + =
" mediump float alen = length(azeq.xyz); \n " ;
if ( hyperbolic ) fmain + =
" azeq *= sinh(alen) / alen; \n "
" azeq[3] = cosh(alen); \n " ;
else if ( sphere ) fmain + =
" azeq *= sin(alen) / alen; \n "
" azeq[3] = cos(alen); \n " ;
else /* euclid */ fmain + =
" azeq[3] = 1; \n " ;
fmain + =
" mediump vec4 nposition = vw * azeq; \n " ;
}
}
else if ( hyperbolic ) {
fmain + =
" mediump float ch = cosh(dist); mediump float sh = sinh(dist); \n "
" mediump vec4 v = position * ch + tangent * sh; \n "
" mediump vec4 ntangent = tangent * ch + position * sh; \n "
" mediump vec4 nposition = v; \n " ;
}
else if ( sphere & & ! stretch : : in ( ) ) {
fmain + =
" mediump float ch = cos(dist); mediump float sh = sin(dist); \n "
" mediump vec4 v = position * ch + tangent * sh; \n "
" mediump vec4 ntangent = tangent * ch - position * sh; \n "
" mediump vec4 nposition = v; \n " ;
}
2022-12-08 18:38:06 +00:00
bool reg = hyperbolic | | sphere | | euclid | | sl2 | | gproduct ;
2021-09-16 18:50:39 +00:00
if ( reg ) {
2022-12-08 18:38:06 +00:00
string s = ( rotspace | | gproduct ) ? " -2 " : " " ;
2021-09-16 18:50:39 +00:00
fmain + =
2021-10-03 08:55:07 +00:00
" mediump float best = " + f_len ( ) + " (nposition); \n "
2021-09-16 18:50:39 +00:00
" for(int i=0; i<sides " + s + " ; i++) { \n "
2021-10-03 08:55:07 +00:00
" mediump float cand = " + f_len ( ) + " ( " + getM ( " walloffset+i " ) + " * nposition); \n "
2021-09-16 18:50:39 +00:00
" if(cand < best) { best = cand; which = i; } \n "
" } \n " ;
if ( rotspace ) fmain + =
" if(which == -1) { \n "
2021-10-03 08:55:07 +00:00
" best = len_rotspace(nposition); \n "
" mediump float cand1 = len_rotspace( " + getM ( " walloffset+sides-2 " ) + " *nposition); \n "
2021-09-16 18:50:39 +00:00
" if(cand1 < best) { best = cand1; which = sides-2; } \n "
2021-10-03 08:55:07 +00:00
" mediump float cand2 = len_rotspace( " + getM ( " walloffset+sides-1 " ) + " *nposition); \n "
2021-09-16 18:50:39 +00:00
" if(cand2 < best) { best = cand2; which = sides-1; } \n "
" } \n " ;
2022-12-08 18:38:06 +00:00
if ( gproduct ) {
2021-09-16 18:50:39 +00:00
fmain + =
" if(nzpos > uPLevel) which = sides-1; \n "
" if(nzpos <-uPLevel) which = sides-2; \n " ;
}
}
if ( nil ) fmain + =
" mediump float rz = (abs(nposition.x) > abs(nposition.y) ? -nposition.x*nposition.y : 0.) + nposition.z; \n " ;
fmain + =
" if(next >= minstep) { \n " ;
string hnilw = to_glsl ( nilv : : nilwidth / 2 ) ;
string hnilw2 = to_glsl ( nilv : : nilwidth * nilv : : nilwidth / 2 ) ;
if ( reg ) fmain + = " if(which != -1) { \n " ;
else if ( asonov ) fmain + =
" if(abs(sp.x) > 1. || abs(sp.y) > 1. || abs(sp.z) > 1.) { \n " ;
else if ( nih ) fmain + =
" if(abs(nposition.x) > uBinaryWidth || abs(nposition.y) > uBinaryWidth || abs(nposition.z) > .5) { \n " ;
else if ( sol ) fmain + =
" if(abs(nposition.x) > uBinaryWidth || abs(nposition.y) > uBinaryWidth || abs(nposition.z) > log(2.)/2.) { \n " ;
else fmain + =
" if(abs(nposition.x) > " + hnilw + " || abs(nposition.y) > " + hnilw + " || abs(rz) > " + hnilw2 + " ) { \n " ;
fmain + =
" next = dist / 2.; continue; \n "
" } \n "
" if(next < maxstep) next = next / 2.; \n "
" } \n "
" else { \n " ;
if ( sn : : in ( ) ) {
if ( asonov ) fmain + =
" if(sp.x > 1.) which = 4; \n "
" if(sp.y > 1.) which = 5; \n "
" if(sp.x <-1.) which = 10; \n "
" if(sp.y <-1.) which = 11; \n "
" if(sp.z > 1.) { \n "
" mediump float best = 999.; \n "
" for(int i=0; i<4; i++) { \n "
2021-10-03 08:55:07 +00:00
" mediump float cand = " + f_len ( ) + " (uStraighten * " + getM ( " i " ) + " * position); \n "
2021-09-16 18:50:39 +00:00
" if(cand < best) { best = cand; which = i;} \n "
" } \n "
" } \n "
" if(sp.z < -1.) { \n "
" mediump float best = 999.; \n "
" for(int i=6; i<10; i++) { \n "
2021-10-03 08:55:07 +00:00
" mediump float cand = " + f_len ( ) + " (uStraighten * " + getM ( " i " ) + " * position); \n "
2021-09-16 18:50:39 +00:00
" if(cand < best) { best = cand; which = i;} \n "
" } \n "
" } \n " ;
else if ( sol & & ! nih ) fmain + =
" if(nposition.x > uBinaryWidth) which = 0; \n "
" if(nposition.x <-uBinaryWidth) which = 4; \n "
" if(nposition.y > uBinaryWidth) which = 1; \n "
" if(nposition.y <-uBinaryWidth) which = 5; \n " ;
if ( nih ) fmain + =
" if(nposition.x > uBinaryWidth) which = 0; \n "
" if(nposition.x <-uBinaryWidth) which = 2; \n "
" if(nposition.y > uBinaryWidth) which = 1; \n "
" if(nposition.y <-uBinaryWidth) which = 3; \n " ;
if ( sol & & nih ) fmain + =
" if(nposition.z > .5) which = nposition.x > 0. ? 5 : 4; \n "
" if(nposition.z <-.5) which = nposition.y > uBinaryWidth/3. ? 8 : nposition.y < -uBinaryWidth/3. ? 6 : 7; \n " ;
if ( nih & & ! sol ) fmain + =
" if(nposition.z > .5) which = 4; \n "
" if(nposition.z < -.5) which = (nposition.y > uBinaryWidth/3. ? 9 : nposition.y < -uBinaryWidth/3. ? 5 : 7) + (nposition.x>0.?1:0); \n " ;
if ( sol & & ! nih & & ! asonov ) fmain + =
" if(nposition.z > log(2.)/2.) which = nposition.x > 0. ? 3 : 2; \n "
" if(nposition.z <-log(2.)/2.) which = nposition.y > 0. ? 7 : 6; \n " ;
}
else if ( nil ) fmain + =
" if(nposition.x > " + hnilw + " ) which = 3; \n "
" if(nposition.x <- " + hnilw + " ) which = 0; \n "
" if(nposition.y > " + hnilw + " ) which = 4; \n "
" if(nposition.y <- " + hnilw + " ) which = 1; \n "
" if(rz > " + hnilw2 + " ) which = 5; \n "
" if(rz <- " + hnilw2 + " ) which = 2; \n " ;
fmain + =
" next = maxstep; \n "
" } \n " ;
if ( use_christoffel ) fmain + =
" tangent = tangent + (acc1+2.*acc2+2.*acc3+acc4)/(6.*dist); \n " ;
else if ( nil & & ! eyes ) fmain + =
" tangent = translatev(position, xt); \n " ;
else if ( ! eyes )
fmain + =
" tangent = ntangent; \n " ;
if ( ! eyes ) fmain + =
" position = nposition; \n " ;
else fmain + = " vec4 position = nposition; \n " ;
if ( ( stretch : : in ( ) | | sl2 ) & & use_christoffel ) {
fmain + =
" tangent = s_itranslate(toOrig * position) * toOrig * tangent; \n "
" tangent[3] = 0.; \n " ;
if ( stretch : : mstretch )
fmain + =
" float nvelsquared = dot(tangent.xyz, (uATOI * tangent).xyz); \n " ;
else
fmain + =
" float nvelsquared = tangent.x * tangent.x + tangent.y * tangent.y + "
+ to_glsl ( stretch : : squared ( ) ) + " * tangent.z * tangent.z; \n " ;
fmain + =
" tangent /= sqrt(nvelsquared); \n "
" tangent = fromOrig * s_translate(toOrig * position) * tangent; \n " ;
}
}
else fmain + =
" position = position + tangent * dist; \n " ;
}
2021-10-03 08:55:07 +00:00
void raygen : : apply_reflect ( int flat1 , int flat2 ) {
2022-12-08 18:38:06 +00:00
if ( gproduct ) fmain + = " if(reflect && which >= sides-2) { zspeed = -zspeed; continue; } \n " ;
2021-10-07 19:44:29 +00:00
if ( horos ( ) ) {
fmain + =
" if(reflect && (which < " + its ( flat1 ) + " || which >= " + its ( flat2 ) + " )) { \n " ;
if ( hyperbolic ) fmain + =
2021-09-16 18:50:39 +00:00
" mediump float x = -log(position.w - position.x); \n "
2021-10-07 19:44:29 +00:00
" mediump vec4 xtan = xpush_h3(-x) * tangent; \n "
2021-09-16 18:50:39 +00:00
" mediump float diag = (position.y*position.y+position.z*position.z)/2.; \n "
" mediump vec4 normal = vec4(1.-diag, -position.y, -position.z, -diag); \n "
2021-10-07 19:44:29 +00:00
" mediump float mdot = dot(xtan.xyz, normal.xyz) - xtan.w * normal.w; \n " ;
else fmain + =
" mediump float x = -log(position.z - position.x); \n "
" mediump vec4 xtan = xpush_h2(-x) * tangent; \n "
" mediump float diag = position.y*position.y/2.; \n "
" mediump vec4 normal = vec4(1.-diag, -position.y, -diag, 0.); \n "
" mediump float mdot = dot(xtan.xy, normal.xy) - xtan.w * normal.w; \n " ;
fmain + =
2021-09-16 18:50:39 +00:00
" xtan = xtan - normal * mdot * 2.; \n "
2021-10-03 08:55:07 +00:00
" tangent = xpush_h(x) * xtan; \n "
2021-09-16 18:50:39 +00:00
" continue; \n "
" } \n " ;
2021-10-07 19:44:29 +00:00
}
2021-09-16 18:50:39 +00:00
if ( asonov ) {
fmain + =
" if(reflect) { \n "
" if(which == 4 || which == 10) tangent = refl(tangent, position.z, uReflectX); \n "
" else if(which == 5 || which == 11) tangent = refl(tangent, position.z, uReflectY); \n "
" else tangent.z = -tangent.z; \n "
" } \n " ;
fsh + =
" uniform mediump vec4 uReflectX, uReflectY; \n "
" mediump vec4 refl(mediump vec4 t, float z, mediump vec4 r) { \n "
" t.x *= exp(z); t.y /= exp(z); \n "
" t -= dot(t, r) * r; \n "
" t.x /= exp(z); t.y *= exp(z); \n "
" return t; \n "
" } \n " ;
}
else if ( sol & & ! nih & & ! asonov ) fmain + =
" if(reflect) { \n "
" if(which == 0 || which == 4) tangent.x = -tangent.x; \n "
" else if(which == 1 || which == 5) tangent.y = -tangent.y; \n "
" else tangent.z = -tangent.z; \n "
" continue; \n "
" } \n " ;
else if ( nih ) fmain + =
" if(reflect) { \n "
" if(which == 0 || which == 2) tangent.x = -tangent.x; \n "
" else if(which == 1 || which == 3) tangent.y = -tangent.y; \n "
" else tangent.z = -tangent.z; \n "
" continue; \n "
" } \n " ;
else {
fmain + =
" if(reflect) { \n "
" tangent = " + getM ( " uMirrorShift+walloffset+which " ) + " * tangent; \n "
" continue; \n "
" } \n " ;
2021-09-16 20:14:30 +00:00
if ( fsh . find ( " uMirrorShift " ) = = string : : npos )
fsh + = " uniform int uMirrorShift; \n " ;
2021-09-16 18:50:39 +00:00
}
}
2021-09-16 19:30:26 +00:00
void raygen : : emit_intra_portal ( int gid1 , int gid2 ) {
if ( 1 ) {
intra : : resetter ir ;
intra : : switch_to ( gid2 ) ;
}
2022-12-08 18:38:06 +00:00
if ( gproduct ) {
2021-09-30 14:13:47 +00:00
fmain + = " mediump vec4 nposition; \n " ;
fmain + = " if(pconnection.x != .5) { \n " ; // kind != 0
if ( 1 ) {
2021-10-07 23:30:35 +00:00
string fn = bt : : in ( ) ? " to_poco_h2xr_b " : in_h2xe ( ) ? " to_poco_h2xr_h " : " to_poco_s2xr_s " ;
2021-10-09 08:24:35 +00:00
string tmul = " " ;
if ( ! bt : : in ( ) ) {
fmain + =
" mediump mat4 tkt = " + getM ( " mid+1 " ) + " ; \n " ;
tmul = " tkt* " ;
}
2021-09-30 14:13:47 +00:00
fmain + =
" nposition = position + tangent * xspeed * 1e-3; \n "
2021-10-09 08:24:35 +00:00
" position = " + fn + " ( " + tmul + " position); \n "
2021-09-30 14:13:47 +00:00
" position.z = 0.; \n " // zpos - uPLevel;\n"
2021-10-09 08:24:35 +00:00
" nposition = " + fn + " ( " + tmul + " nposition); \n "
2021-09-30 14:13:47 +00:00
" nposition.z = zspeed * 1e-3; \n "
2021-10-02 12:01:22 +00:00
" if(pconnection.y < .5) { nposition.z = -nposition.z; nposition.x = -nposition.x; position.x = -position.x; } \n " ;
2021-09-30 14:13:47 +00:00
}
fmain + = " } else { \n " ;
if ( 1 ) {
string fn = in_h2xe ( ) ? " to_poco_h2xr_e " : " to_poco_s2xr_e " ;
fmain + =
" nposition = position + tangent * xspeed * 1e-3; \n "
" mediump mat4 tkt = " + getM ( " mid+1 " ) + " ; \n "
" position = " + fn + " (tkt * position); \n "
" position.y = zpos; \n "
" nposition = " + fn + " (tkt * nposition); \n "
" nposition.y = zpos + zspeed * 1e-3; \n " ;
}
fmain + = " } \n " ;
2021-09-16 19:30:26 +00:00
}
2021-10-03 08:19:45 +00:00
else if ( hyperbolic & & bt : : in ( ) ) {
fmain + =
" mediump vec4 nposition = position + tangent * 1e-3; \n "
" mediump mat4 tkt = " + getM ( " mid+1 " ) + " ; \n "
2022-05-17 07:41:49 +00:00
" position = deparabolici13(position); \n "
" nposition = deparabolici13(nposition); \n "
2021-10-07 11:24:39 +00:00
" position = tkt * position; \n "
" nposition = tkt * nposition; \n "
" position.z *= exp(position.y); \n "
" nposition.z *= exp(nposition.y); \n " ;
2021-10-03 16:06:15 +00:00
}
else if ( sol ) {
fmain + =
" mediump vec4 nposition = position + tangent * 1e-3; \n "
" mediump mat4 tkt = " + getM ( " mid+1 " ) + " ; \n "
" position = tkt * position; \n "
2021-10-07 11:24:39 +00:00
" nposition = tkt * nposition; \n "
" position.z *= exp(-position.y); \n "
" nposition.z *= exp(-nposition.y); \n " ;
2021-10-03 08:19:45 +00:00
}
2021-09-16 19:30:26 +00:00
else {
fmain + =
" mediump vec4 nposition = position + tangent * 1e-3; \n "
" mediump mat4 tkt = " + getM ( " mid+1 " ) + " ; \n " ;
if ( hyperbolic ) fmain + =
" position = to_poco_h3(tkt * position); \n "
" nposition = to_poco_h3(tkt * nposition); \n " ;
else if ( sphere ) fmain + =
" position = to_poco_s3(tkt * position); \n "
" nposition = to_poco_s3(tkt * nposition); \n " ;
else fmain + =
" position = tkt * position; \n "
" nposition = tkt * nposition; \n " ;
}
fmain + =
" mediump mat4 m = " + getM ( " mid " ) + " ; \n "
" position = m * position; \n "
" nposition = m * nposition; \n " ;
intra : : resetter ir ;
intra : : switch_to ( gid2 ) ;
2022-12-08 18:38:06 +00:00
if ( gproduct ) {
2021-09-30 14:13:47 +00:00
fmain + = " if(pconnection.z != .5) { \n " ; // kind != 0
if ( 1 ) {
string sgn = in_h2xe ( ) ? " - " : " + " ;
2021-10-09 08:24:35 +00:00
string fn = bt : : in ( ) ? " from_poco_h2xr_b " : in_h2xe ( ) ? " itkt*from_poco_h2xr_h " : " itkt*from_poco_s2xr_s " ;
2021-09-30 14:13:47 +00:00
fmain + =
2021-10-02 12:01:22 +00:00
" if(pconnection.w < .5) { position.z = -position.z; nposition.z = -nposition.z; nposition.x = -nposition.x; position.x = -position.x; } \n "
2021-09-30 14:13:47 +00:00
" zspeed = (nposition.z - position.z) * 1e3; \n "
" zpos = position.z + (pconnection.w - .5) * 16.; \n "
2021-10-08 09:27:23 +00:00
" mediump mat4 itkt = " + getM ( " mid+2 " ) + " ; \n "
2021-10-09 08:24:35 +00:00
" position = " + fn + " (position); \n "
" nposition = " + fn + " (nposition); \n "
2021-09-30 14:13:47 +00:00
" tangent = (nposition - position) * 1e3; \n "
" mediump float pnorm = tangent.z * position.z " + sgn + " tangent.x * position.x " + sgn + " tangent.y * position.y; \n "
" tangent -= position * pnorm; \n "
" xspeed = sqrt(tangent.x * tangent.x + tangent.y * tangent.y " + sgn + " tangent.z * tangent.z); \n "
" tangent /= xspeed; \n " ;
}
fmain + = " } else { \n " ;
if ( 1 ) {
string sgn = in_h2xe ( ) ? " - " : " + " ;
string fn = in_h2xe ( ) ? " from_poco_h2xr_e " : " from_poco_s2xr_e " ;
fmain + =
" mediump mat4 itkt = " + getM ( " mid+2 " ) + " ; \n " ;
fmain + =
" zpos = position.y; \n "
" zspeed = (nposition.y - zpos) * 1e3; \n "
" position = itkt * " + fn + " (position); \n "
" nposition = itkt * " + fn + " (nposition); \n "
" tangent = (nposition - position) * 1e3; \n "
" mediump float pnorm = tangent.z * position.z " + sgn + " dot(position.xy, tangent.xy); \n "
" tangent -= position * pnorm; \n "
" xspeed = sqrt(dot(tangent.xy, tangent.xy) " + sgn + " tangent.z * tangent.z); \n "
" tangent /= xspeed; \n "
" mediump float l = xspeed*xspeed+zspeed*zspeed; \n "
" xspeed /= sqrt(l); zspeed /= sqrt(l); \n " ;
}
fmain + = " } \n " ;
2021-09-16 19:30:26 +00:00
}
2021-10-03 16:06:15 +00:00
else if ( sol ) {
fmain + =
" mediump mat4 itkt = " + getM ( " mid+2 " ) + " ; \n " ;
fmain + =
2021-10-07 11:24:39 +00:00
" position.z *= exp(position.y); \n "
" nposition.z *= exp(nposition.y); \n "
2021-10-03 16:06:15 +00:00
" position = itkt * position; \n "
" nposition = itkt * nposition; \n "
" tangent = (nposition - position) * 1e3; \n "
" float next = maxstep; \n " ;
}
2021-09-16 19:30:26 +00:00
else {
fmain + =
" mediump mat4 itkt = " + getM ( " mid+2 " ) + " ; \n " ;
2021-10-03 08:19:45 +00:00
if ( hyperbolic | | sphere ) {
string sgn = hyperbolic ? " - " : " + " ;
if ( hyperbolic & & bt : : in ( ) ) {
fmain + =
2021-10-07 11:24:39 +00:00
" position.z *= exp(-position.y); \n "
" nposition.z *= exp(-nposition.y); \n "
" position = itkt * position; \n "
" nposition = itkt * nposition; \n "
2022-05-17 07:41:49 +00:00
" position = enparabolici13(position); \n "
" nposition = enparabolici13(nposition); \n " ;
2021-10-03 08:19:45 +00:00
}
else {
string he = hyperbolic ? " from_poco_h3 " : " from_poco_s3 " ;
fmain + =
2021-09-16 19:30:26 +00:00
" position = itkt * " + he + " (position); \n "
2021-10-03 08:19:45 +00:00
" nposition = itkt * " + he + " (nposition); \n " ;
}
fmain + =
2021-09-16 19:30:26 +00:00
" tangent = (nposition - position) * 1e3; \n "
" mediump float pnorm = position.w * position.w " + sgn + " dot(position.xyz, position.xyz); \n "
" position /= sqrt(pnorm); \n "
" pnorm = tangent.w * position.w " + sgn + " dot(position.xyz, tangent.xyz); \n "
" tangent -= position * pnorm; \n "
" mediump float xspeed = sqrt(dot(tangent.xyz, tangent.xyz) " + sgn + " tangent.w * tangent.w); \n "
" tangent /= xspeed; \n " ;
2021-10-03 08:19:45 +00:00
}
2021-09-16 19:30:26 +00:00
else fmain + =
" position = itkt * position; \n "
" nposition = itkt * nposition; \n "
" tangent = (nposition - position) * 1e3; \n "
" tangent /= dot(tangent.xyz, tangent.xyz); \n " ;
}
}
void raygen : : emit_iterate ( int gid1 ) {
2021-09-16 18:50:39 +00:00
using glhr : : to_glsl ;
2022-12-08 18:38:06 +00:00
if ( intra : : in & & gproduct )
2021-09-16 19:30:26 +00:00
fmain + = " const mediump float uPLevel = " + to_glsl ( cgi . plevel / 2 ) + " ; \n " ;
2021-10-03 08:55:07 +00:00
int flat1 = 0 , flat2 = deg ;
2022-12-08 18:38:06 +00:00
if ( gproduct | | rotspace ) flat2 - = 2 ;
2021-10-03 08:55:07 +00:00
# if CAP_BT
2021-10-07 19:44:29 +00:00
if ( horos ( ) ) {
2021-10-03 08:55:07 +00:00
if ( intra : : in )
fmain + = " mediump float uBLevel = " + to_glsl ( log ( bt : : expansion ( ) ) / 2 ) + " ; \n " ;
else
fsh + = " uniform mediump float uBLevel; \n " ;
flat1 = bt : : dirs_outer ( ) ;
flat2 - = bt : : dirs_inner ( ) ;
}
# endif
2021-09-16 18:50:39 +00:00
fmain + =
" mediump float dist = 100.; \n " ;
fmain + =
" int which = -1; \n " ;
2021-10-09 15:04:54 +00:00
if ( intra : : in ) {
if ( hyperbolic ) fmain + = " iter += 44; \n " ;
else if ( ! sol ) fmain + = " iter += 4 ; \ n " ;
}
2021-09-16 19:30:26 +00:00
2021-09-16 18:50:39 +00:00
if ( in_e2xe ( ) & & ! eyes ) fmain + = " tangent.w = position.w = 0.; \n " ;
if ( IN_ODS ) fmain + =
" if(go == 0.) { \n "
2021-10-03 08:55:07 +00:00
" mediump float best = " + f_len ( ) + " (position); \n "
2021-09-16 18:50:39 +00:00
" for(int i=0; i<sides; i++) { \n "
2021-10-03 08:55:07 +00:00
" mediump float cand = " + f_len ( ) + " ( " + getM ( " i " ) + " * position); \n "
2021-09-16 18:50:39 +00:00
" if(cand < best - .001) { dist = 0.; best = cand; which = i; } \n "
" } \n "
" } \n " ;
2021-10-03 08:55:07 +00:00
compute_which_and_dist ( flat1 , flat2 ) ;
2021-09-16 18:50:39 +00:00
vid . fixed_yz = false ;
// shift d units
if ( use_reflect ) fmain + =
" bool reflect = false; \n " ;
move_forward ( ) ;
if ( ! eyes ) {
if ( hyperbolic ) fmain + =
" position /= sqrt(position.w*position.w - dot(position.xyz, position.xyz)); \n "
" tangent -= dot(vec4(-position.xyz, position.w), tangent) * position; \n "
" tangent /= sqrt(dot(tangent.xyz, tangent.xyz) - tangent.w*tangent.w); \n " ;
if ( in_h2xe ( ) ) fmain + =
" position /= sqrt(position.z*position.z - dot(position.xy, position.xy)); \n "
" tangent -= dot(vec3(-position.xy, position.z), tangent.xyz) * position; \n "
" tangent /= sqrt(dot(tangent.xy, tangent.xy) - tangent.z*tangent.z); \n " ;
}
2021-10-07 19:44:29 +00:00
if ( horos ( ) ) {
string w20 , w21 ;
if ( in_h2xe ( ) & & hybrid : : underlying = = gBinary4 ) {
w21 = " for(int i=0; i<2; i++) { \n " ;
w20 = " int i=3; { \n " ;
}
2021-10-07 20:26:41 +00:00
else if ( in_h2xe ( ) & & hybrid : : underlying = = gTernary ) {
w21 = " for(int i=0; i<3; i++) { \n " ;
w20 = " int i=4; { \n " ;
}
else if ( in_h2xe ( ) & & hybrid : : underlying = = gBinaryTiling ) {
w20 = " int i = sides == 8 ? 5 : position.y < 0. ? 5 : 6; { \n " ;
w21 = " for(int i=1; i<4; i++) { \n " ;
}
2021-10-07 19:44:29 +00:00
else {
w20 = " for(int i= " + its ( flat2 ) + " ; i< " + its ( S7 ) + " ; i++) { \n " ;
w21 = " for(int i=0; i< " + its ( flat1 ) + " ; i++) { \n " ;
}
2021-09-16 18:50:39 +00:00
fmain + =
" if(which == 20) { \n "
" mediump float best = 999.; \n "
2021-10-07 19:44:29 +00:00
+ w20 +
2021-10-07 21:16:16 +00:00
" mediump float cand = " + f_len_prod ( ) + " ( " + getM ( " walloffset+i " ) + " * position); \n "
2021-09-16 18:50:39 +00:00
" if(cand < best) { best = cand; which = i; } \n "
" } \n "
" } \n "
" if(which == 21) { \n "
" mediump float best = 999.; \n "
2021-10-07 19:44:29 +00:00
+ w21 +
2021-10-07 21:16:16 +00:00
" mediump float cand = " + f_len_prod ( ) + " ( " + getM ( " walloffset+i " ) + " * position); \n "
2021-09-16 18:50:39 +00:00
" if(cand < best) { best = cand; which = i; } \n "
" } \n "
" } \n " ;
}
if ( volumetric : : on ) fmain + =
" if(dist > 0. && go < " + to_glsl ( hard_limit ) + " ) { \n "
" if(dist > " + to_glsl ( hard_limit ) + " - go) dist = " + to_glsl ( hard_limit ) + " - go; \n "
" mediump vec4 col = texture2D(tVolumetric, cid); \n "
" mediump float factor = col.w; col.w = 1.; \n "
" mediump float frac = exp(-(factor + 1. / uExpDecay) * dist); \n "
" gl_FragColor += left * (1.-frac) * col; \n "
" left *= frac; \n "
" } \n ; " ;
fmain + = " go = go + dist; \n " ;
fmain + = " if(which == -1) continue; \n " ;
2022-12-08 18:38:06 +00:00
if ( gproduct & & eyes ) fmain + = " position.w = -nzpos; \n " ;
else if ( gproduct ) fmain + = " position.w = -zpos ; \ n " ;
2021-09-16 18:50:39 +00:00
if ( reg3 : : ultra_mirror_in ( ) ) fmain + =
" if(which >= " + its ( S7 ) + " ) { "
" tangent = " + getM ( " which " ) + " * tangent; \n "
" continue; \n "
" } \n " ;
// apply wall color
fmain + =
" mediump vec2 u = cid + vec2(float(which) / float(uLength), 0); \n "
" mediump vec4 col = texture2D(tWallcolor, u); \n "
" if(col[3] > 0.0) { \n " ;
if ( eyes )
fmain + = " mediump float gou = go / uAbsUnit; \n " ;
else
fmain + = " mediump float gou = go; \n " ;
if ( hard_limit < NO_LIMIT )
fmain + = " if(gou > " + to_glsl ( hard_limit ) + " ) { gl_FragDepth = 1.; return; } \n " ;
if ( ! ( levellines & & disable_texture ) ) {
fmain + = " mediump vec4 pos = position; \n " ;
if ( nil ) fmain + = " if(which == 2 || which == 5) pos.z = 0.; \n " ;
else if ( hyperbolic & & bt : : in ( ) ) fmain + =
2022-05-17 07:41:49 +00:00
" pos = deparabolici13(pos); \n "
2021-10-07 16:23:27 +00:00
" pos.xyz = pos.zxy; \n " ;
2021-09-16 18:50:39 +00:00
else if ( hyperbolic | | sphere ) fmain + =
" pos /= pos.w; \n " ;
2022-12-08 18:38:06 +00:00
else if ( gproduct & & bt : : in ( ) ) fmain + =
2021-10-07 19:44:29 +00:00
" pos.xy = deparabolic12(pos).xy; \n "
" pos.z = -pos.w; pos.w = 0.; \n "
;
2022-12-08 18:38:06 +00:00
else if ( gproduct ) fmain + =
2021-10-07 19:44:29 +00:00
" pos = vec4(pos.x/pos.z, pos.y/pos.z, -pos.w, 0); \n " ;
2021-09-16 18:50:39 +00:00
fmain + =
" mediump vec2 inface = map_texture(pos, which+walloffset); \n "
" mediump vec3 tmap = texture2D(tTextureMap, u).rgb; \n "
" if(tmap.z == 0.) col.xyz *= min(1., (1.-inface.x)/ tmap.x); \n "
" else { \n "
" mediump vec2 inface2 = tmap.xy + tmap.z * inface; \n "
" col.xyz *= texture2D(tTexture, inface2).rgb; \n "
" } \n " ;
}
if ( volumetric : : on )
fmain + = " mediump float d = uExpStart * exp(-gou / uExpDecay); \n " ;
else
fmain + =
" mediump float d = max(1. - gou / uLinearSightRange, uExpStart * exp(-gou / uExpDecay)); \n " ;
if ( ! volumetric : : on ) fmain + =
" col.xyz = col.xyz * d + uFogColor.xyz * (1.-d); \n " ;
if ( nil ) fmain + =
" if(abs(abs(position.x)-abs(position.y)) < .005) col.xyz /= 2.; \n " ;
if ( use_reflect ) fmain + =
" if(col.w == 1.) { \n "
" col.w = " + to_glsl ( 1 - reflect_val ) + " ; \n "
" reflect = true; \n "
" } \n " ;
ld vnear = glhr : : vnear_default ;
ld vfar = glhr : : vfar_default ;
fmain + =
" gl_FragColor.xyz += left * col.xyz * col.w; \n " ;
if ( use_reflect ) fmain + =
" if(reflect && depthtoset) { \n " ;
else fmain + =
" if(col.w == 1.) { \n " ;
2022-03-27 17:43:01 +00:00
if ( hyperbolic & & ! eyes & & ! intra : : in ) {
fmain + =
" mediump vec4 t = at0 * tanh(go); \n "
2021-09-16 18:50:39 +00:00
" t.w = 1.; \n " ;
2022-03-27 17:43:01 +00:00
fmain + = " gl_FragColor.xyz *= 0.9999 + 0.0001 * t.z; \n " ;
fmain + = " gl_FragColor.xyz /= 0.9999 + 0.0001 * t.z; \n " ;
}
else {
fmain + =
" mediump vec4 t = at0 * go; \n "
" t.w = 1.; \n " ;
}
2021-09-16 18:50:39 +00:00
if ( levellines ) {
2021-10-03 16:06:27 +00:00
if ( hyperbolic & & ! eyes & & ! intra : : in )
2022-03-27 17:42:04 +00:00
fmain + = " gl_FragColor.xyz *= 0.5 + 0.5 * cos(t.z * uLevelLines * 2. * PI); \n " ;
2021-09-16 18:50:39 +00:00
else
2021-10-03 16:06:27 +00:00
fmain + = " gl_FragColor.xyz *= 0.5 + 0.5 * cos(go * uLevelLines * 2. * PI); \n " ;
if ( fsh . find ( " uLevelLines " ) = = string : : npos )
fsh + = " uniform mediump float uLevelLines; \n " ;
2021-09-16 18:50:39 +00:00
}
if ( panini_alpha )
fmain + = panini_shader ( ) ;
else if ( stereo_alpha )
fmain + = stereo_shader ( ) ;
# ifndef GLES_ONLY
fmain + =
" gl_FragDepth = ( " + to_glsl ( - vnear - vfar ) + " +t.w* " + to_glsl ( 2 * vnear * vfar ) + " /t.z)/ " + to_glsl ( vnear - vfar ) + " ; \n "
" gl_FragDepth = (gl_FragDepth + 1.) / 2.; \n " ;
# endif
if ( ! use_reflect ) fmain + =
" return; \n " ;
else fmain + =
" depthtoset = false; \n " ;
fmain + =
" } \n "
" left *= (1. - col.w); \n "
" } \n " ;
if ( use_reflect )
2021-10-03 08:55:07 +00:00
apply_reflect ( flat1 , flat2 ) ;
2021-09-16 18:50:39 +00:00
// next cell
fmain + =
" mediump vec4 connection = texture2D(tConnections, u); \n "
" cid = connection.xy; \n " ;
if ( many_cell_types )
fmain + = " int nwalloffset = int(connection.w * " + to_glsl ( max_wall_offset ) + " ); \n " ;
fmain + = " int mid = int(connection.z * 1024.); \n " ;
string no_intra_portal = " " ;
no_intra_portal + =
" mediump mat4 m = " + getM ( " mid " ) + " * " + getM ( " walloffset+which " ) + " ; \n " ;
if ( eyes )
no_intra_portal + = " vw = m * vw; \n " ;
else no_intra_portal + =
" position = m * position; \n "
" tangent = m * tangent; \n " ;
2022-12-08 18:38:06 +00:00
if ( gproduct ) no_intra_portal =
2021-09-16 18:50:39 +00:00
" if(which == sides-2) { zpos += uPLevel+uPLevel; } \n "
" else if(which == sides-1) { zpos -= uPLevel+uPLevel; } \n "
" else { \n " + no_intra_portal + " } \n " ;
if ( stretch : : mstretch ) no_intra_portal + =
" m = s_itranslate(m*vec4(0,0,0,1)) * m; "
" fromOrig = m * fromOrig; \n "
" m[0][1] = -m[0][1]; m[1][0] = -m[1][0]; \n " // inverse
" toOrig = toOrig * m; \n " ;
2021-09-16 19:30:26 +00:00
if ( ! intra : : in ) fmain + = no_intra_portal ;
else {
2021-10-03 08:55:07 +00:00
if ( intra : : in ) {
int q = isize ( intra : : data ) ;
fmain + = " mediump vec4 pconnection = texture2D(tPortalConnections, u); \n " ;
fmain + = " if(pconnection.x == 0.) { \n " ;
fmain + = no_intra_portal ;
fmain + = " } else \n " ;
for ( int gid2 = 0 ; gid2 < q ; gid2 + + ) {
if ( gid2 = = q - 1 )
fmain + = " { \n " ;
else {
fmain + = " if(nwalloffset < " + its ( intra : : data [ gid2 + 1 ] . wallindex ) + " ) { \n " ;
}
emit_intra_portal ( gid1 , gid2 ) ;
if ( gid2 = = q - 1 )
fmain + = " } \n " ;
else
fmain + = " } else \n " ;
}
}
else
fmain + = no_intra_portal ;
}
if ( many_cell_types ) {
fmain + =
" walloffset = nwalloffset; \n "
" sides = int(connection.w * " + to_glsl ( max_wall_offset * max_celltype ) + " ) - " + its ( max_celltype ) + " * walloffset; \n " ;
// fmain += "if(sides != 8) { gl_FragColor = vec4(.5,float(sides)/8.,.5,1); return; }";
}
}
void enable_raycaster ( ) {
using glhr : : to_glsl ;
auto state = raycaster_state ( ) ;
if ( state ! = saved_state ) {
reset_raycaster ( ) ;
saved_state = state ;
}
last_geometry = geometry ;
if ( ! our_raycaster ) {
auto & g = our_raygen ;
g . create ( ) ;
g . fsh + = g . fmain ;
callhooks ( hooks_rayshader , g . vsh , g . fsh ) ;
our_raycaster = make_shared < raycaster > ( g . vsh , g . fsh ) ;
}
full_enable ( our_raycaster ) ;
}
void raygen : : add_functions ( ) {
2021-10-07 19:44:29 +00:00
add_if ( " xpush_h3 " ,
2021-10-03 08:55:07 +00:00
2021-10-07 19:44:29 +00:00
" mediump mat4 xpush_h3(float x) { return mat4( "
2021-10-03 08:55:07 +00:00
" cosh(x), 0., 0., sinh(x), \n "
" 0., 1., 0., 0., \n "
" 0., 0., 1., 0., \n "
" sinh(x), 0., 0., cosh(x) "
" );} \n " ) ;
2022-12-15 20:17:08 +00:00
add_if ( " zpush_h3 " ,
" mediump mat4 zpush_h3(float x) { return mat4( "
" 1., 0., 0., 0., \n "
" 0., 1., 0., 0., \n "
" 0., 0., cosh(x), sinh(x), \n "
" 0., 0., sinh(x), cosh(x) "
" );} \n " ) ;
2021-10-07 19:44:29 +00:00
add_if ( " xpush_h2 " ,
" mediump mat4 xpush_h2(float x) { return mat4( "
" cosh(x), 0., sinh(x), 0., \n "
" 0., 1., 0., 0., \n "
" sinh(x), 0., cosh(x), 0., \n "
" 0., 0., 0., 1. "
" );} \n " ) ;
2021-10-03 08:55:07 +00:00
add_if ( " xpush_s " ,
2021-10-07 19:44:29 +00:00
" mediump mat4 xpush_s(float x) { return mat4( "
2021-10-03 08:55:07 +00:00
" cos(x), 0., 0., sin(x), \n "
" 0., 1., 0., 0., \n "
" 0., 0., 1., 0., \n "
" -sin(x), 0., 0., cos(x) "
" );} \n "
) ;
add_if ( " xzspin " ,
" mediump mat4 xzspin(float x) { return mat4( "
" cos(x), 0., sin(x), 0., \n "
" 0., 1., 0., 0., \n "
" -sin(x), 0., cos(x), 0., \n "
" 0., 0., 0., 1. "
" );} \n " ) ;
add_if ( " yzspin " ,
" mediump mat4 yzspin(float x) { return mat4( "
" 1., 0., 0., 0., \n "
" 0., cos(x), sin(x), 0., \n "
" 0., -sin(x), cos(x), 0., \n "
" 0., 0., 0., 1. "
" );} \n "
) ;
2022-05-17 07:41:49 +00:00
/* note: this is called deparabolici13 because it orders the coordinates differently from deparabolic13 */
add_if ( " deparabolici13 " ,
" mediump vec4 deparabolici13(mediump vec4 h) { \n "
2021-10-03 08:19:45 +00:00
" h /= (1. + h[3]); \n "
" h[0] -= 1.; \n "
" h /= h.x*h.x + h.y*h.y + h.z * h.z; \n "
" h[0] += .5; \n "
" mediump vec4 res; \n "
2021-10-07 11:24:39 +00:00
" res.x = h.y * 2.; \n "
" res.y = h.z * 2.; \n "
" res.z = (log(2.) + log(-h.x)); \n "
2021-10-03 08:19:45 +00:00
" res.w = 1.; \n "
" return res; \n "
2021-10-03 08:55:07 +00:00
" } \n \n " ) ;
2021-10-03 08:19:45 +00:00
2022-05-17 07:41:49 +00:00
/* note: this is called eeparabolici13 because it orders the coordinates differently from enparabolic13 */
add_if ( " enparabolici13 " ,
" mediump vec4 enparabolici13(mediump vec4 h) { \n "
2021-10-03 08:19:45 +00:00
" mediump vec4 res; \n "
" float diag = (h.x*h.x + h.y*h.y)/2.; \n "
2021-10-03 09:16:15 +00:00
" res.x = sinh(h.z) + diag * exp(-h.z); \n "
" res.y = h.x * exp(-h.z); \n "
" res.z = h.y * exp(-h.z); \n "
" res.w = cosh(h.z) + diag * exp(-h.z); \n "
2021-10-03 08:19:45 +00:00
" return res; \n "
2021-10-03 08:55:07 +00:00
" } \n \n " ) ;
2021-10-03 08:19:45 +00:00
2021-10-03 08:55:07 +00:00
add_if ( " to_poco_h3 " ,
2021-09-16 19:30:26 +00:00
" mediump vec4 to_poco_h3(mediump vec4 pos) { \n "
2022-02-26 08:49:46 +00:00
" pos = pos / pos[3]; \n "
" pos[2] /= sqrt(1.-pos.x*pos.x-pos.y*pos.y); \n "
" return pos; \n "
2021-10-03 08:55:07 +00:00
" } \n \n " ) ;
2021-09-16 19:30:26 +00:00
2021-10-03 08:55:07 +00:00
add_if ( " from_poco_h3 " ,
2021-09-16 19:30:26 +00:00
" mediump vec4 from_poco_h3(mediump vec4 pos) { \n "
2022-02-26 08:49:46 +00:00
" pos[2] *= sqrt(1.-pos.x*pos.x-pos.y*pos.y); \n "
2021-09-16 19:30:26 +00:00
" float s = 1. - dot(pos.xyz, pos.xyz); \n "
" return pos / sqrt(s); \n "
2021-10-03 08:55:07 +00:00
" } \n \n " ) ;
2021-09-16 19:30:26 +00:00
2021-10-03 08:55:07 +00:00
add_if ( " to_poco_s3 " ,
2021-09-16 19:30:26 +00:00
" mediump vec4 to_poco_s3(mediump vec4 pos) { \n "
2022-02-26 08:49:46 +00:00
" pos = pos / pos[3]; \n "
" pos[2] /= sqrt(1.+pos.x*pos.x+pos.y*pos.y); \n "
" return pos; \n "
2021-10-03 08:55:07 +00:00
" } \n \n " ) ;
2021-09-16 19:30:26 +00:00
2021-10-03 08:55:07 +00:00
add_if ( " from_poco_s3 " ,
2021-09-16 19:30:26 +00:00
" mediump vec4 from_poco_s3(mediump vec4 pos) { \n "
2022-02-26 08:49:46 +00:00
" pos[2] *= sqrt(1.+pos.x*pos.x+pos.y*pos.y); \n "
2021-09-16 19:30:26 +00:00
" float s = 1. + dot(pos.xyz, pos.xyz); \n "
" return pos / sqrt(s); \n "
2021-10-03 08:55:07 +00:00
" } \n \n " ) ;
2021-09-16 19:30:26 +00:00
2021-10-03 08:55:07 +00:00
add_if ( " from_poco_h2xr_h " ,
2021-09-16 19:30:26 +00:00
" mediump vec4 from_poco_h2xr_h(mediump vec4 pos) { \n "
" float s = 1. - pos.x*pos.x - pos.y * pos.y; \n "
" pos.z = 1.; \n "
" return pos / sqrt(s); \n "
2021-10-03 08:55:07 +00:00
" } \n \n " ) ;
2021-09-16 19:30:26 +00:00
2021-10-03 08:55:07 +00:00
add_if ( " to_poco_h2xr_h " ,
2021-09-16 19:30:26 +00:00
" mediump vec4 to_poco_h2xr_h(mediump vec4 pos) { \n "
" pos /= pos[2]; \n "
" pos.w = 1.; \n "
" return pos; \n "
2021-10-03 08:55:07 +00:00
" } \n \n " ) ;
2021-09-16 19:30:26 +00:00
2021-10-07 23:30:35 +00:00
add_if ( " from_poco_h2xr_b " ,
" mediump vec4 from_poco_h2xr_b(mediump vec4 pos) { \n "
" return enparabolic12(pos); \n "
" } \n \n " ) ;
add_if ( " to_poco_h2xr_b " ,
" mediump vec4 to_poco_h2xr_b(mediump vec4 pos) { \n "
" pos /= sqrt(pos.z*pos.z-pos.x*pos.x-pos.y*pos.y); \n "
" pos = deparabolic12(pos); \n "
" return pos; \n "
" } \n \n " ) ;
2021-10-03 08:55:07 +00:00
add_if ( " from_poco_h2xr_e " ,
2021-09-16 19:30:26 +00:00
" mediump vec4 from_poco_h2xr_e(mediump vec4 pos) { \n "
2022-02-26 08:48:49 +00:00
" return vec4(sinh(pos[2]), sinh(pos[0]) * cosh(pos[2]), cosh(pos[0]) * cosh(pos[2]), 0); \n "
2021-10-03 08:55:07 +00:00
" } \n \n " ) ;
2021-09-16 19:30:26 +00:00
2021-10-03 08:55:07 +00:00
add_if ( " to_poco_h2xr_e " ,
2021-09-16 19:30:26 +00:00
" mediump vec4 to_poco_h2xr_e(mediump vec4 pos) { \n "
2022-02-26 08:48:49 +00:00
" mediump float x = asinh(pos[0]); \n "
" return vec4(asinh(pos[1] / cosh(x)), 0, x, 1); \n "
2021-10-03 08:55:07 +00:00
" } \n \n " ) ;
add_if ( " from_poco_s2xr_s " ,
2021-09-16 19:30:26 +00:00
" mediump vec4 from_poco_s2xr_s(mediump vec4 pos) { \n "
" float s = 1. + pos.x*pos.x + pos.y * pos.y; \n "
" pos.z = 1.; \n "
" return pos / sqrt(s); \n "
2021-10-03 08:55:07 +00:00
" } \n \n " ) ;
2021-09-16 19:30:26 +00:00
2021-10-03 08:55:07 +00:00
add_if ( " to_poco_s2xr_s " ,
2021-09-16 19:30:26 +00:00
" mediump vec4 to_poco_s2xr_s(mediump vec4 pos) { \n "
" pos /= pos[2]; \n "
" pos.w = 1.; \n "
" return pos; \n "
2021-10-03 08:55:07 +00:00
" } \n \n " ) ;
2021-09-16 19:30:26 +00:00
2021-10-03 08:55:07 +00:00
add_if ( " from_poco_s2xr_e " ,
2021-09-16 19:30:26 +00:00
" mediump vec4 from_poco_s2xr_e(mediump vec4 pos) { \n "
2022-02-26 08:48:49 +00:00
" return vec4(sin(pos[2]), sin(pos[0]) * cos(pos[2]), cos(pos[0]) * cos(pos[2]), 0); \n "
2021-10-03 08:55:07 +00:00
" } \n \n " ) ;
2021-09-16 19:30:26 +00:00
2021-10-03 08:55:07 +00:00
add_if ( " to_poco_s2xr_e " ,
2021-09-16 19:30:26 +00:00
" mediump vec4 to_poco_s2xr_e(mediump vec4 pos) { \n "
2022-02-26 08:48:49 +00:00
" mediump float x = asin_clamp(pos[0]); \n "
" return vec4(asin_clamp(pos[1] / cos(x)), 0, x, 1); \n "
2021-10-03 08:55:07 +00:00
" } \n \n " ) ;
2021-09-16 19:30:26 +00:00
2021-10-07 23:30:35 +00:00
add_if ( " deparabolic12 " ,
" mediump vec4 deparabolic12(mediump vec4 h) { \n "
" h /= (1. + h.z); \n "
" h[0] -= 1.; \n "
" h /= h.x*h.x + h.y*h.y; \n "
" h[0] += .5; \n "
" mediump vec4 res; \n "
" res.x = (log(2.) + log(-h.x)); \n "
" res.y = h.y * 2.; \n "
" res.w = 1.; \n "
" return res; \n "
" } \n \n " ) ;
add_if ( " enparabolic12 " ,
" mediump vec4 enparabolic12(mediump vec4 h) { \n "
" mediump vec4 res; \n "
" float diag = h.y*h.y /2.; \n "
" res.x = sinh(h.x) + diag * exp(-h.x); \n "
" res.y = h.y * exp(-h.x); \n "
" res.z = cosh(h.x) + diag * exp(-h.x); \n "
" return res; \n "
" } \n \n " ) ;
2021-10-03 08:55:07 +00:00
add_if ( " len_rotspace " , " mediump float len_rotspace(vec4 h) { return 1. - h[3]; } \n " ) ;
2021-09-30 14:13:47 +00:00
2021-10-07 19:44:29 +00:00
add_if ( " len_h3 " , " mediump float len_h3(mediump vec4 x) { return x[3]; } \n " ) ;
2021-10-07 16:23:40 +00:00
add_if ( " len_sr " , " mediump float len_sr(mediump vec4 x) { return 1.+x.x*x.x+x.y*x.y-x.z*x.z-x.w*x.w; } \n " ) ;
add_if ( " len_sl2 " , " mediump float len_sl2(mediump vec4 x) { return 1.+x.x*x.x+x.y*x.y; } \n " ) ;
2021-10-07 19:44:29 +00:00
add_if ( " len_s3 " , " mediump float len_s3(mediump vec4 x) { return 1.-x[3]; } \n " ) ;
2021-10-07 16:23:40 +00:00
add_if ( " len_x " , " mediump float len_x(mediump vec4 x) { return length(x.xyz); } \n " ) ;
2021-10-07 19:44:29 +00:00
add_if ( " len_h2 " , " mediump float len_h2(mediump vec4 x) { return x[2]; } \n " ) ;
add_if ( " len_s2 " , " mediump float len_s2(mediump vec4 x) { return 1.-x[2]; } \n " ) ;
add_if ( " len_e2 " , " mediump float len_e2(mediump vec4 x) { return length(x.xy); } \n " ) ;
2021-09-16 18:50:39 +00:00
}
2021-10-09 09:11:31 +00:00
void raygen : : emit_raystarter ( ) {
2022-12-08 18:38:06 +00:00
if ( gproduct ) {
2021-10-09 09:11:31 +00:00
string sgn = in_h2xe ( ) ? " - " : " + " ;
fmain + =
" position = vw * vec4(0., 0., 1., 0.); \n "
" mediump vec4 at1 = uLP * at0; \n " ;
if ( in_e2xe ( ) ) fmain + =
" zpos = log(position.z); \n " ;
else fmain + =
" zpos = log(position.z*position.z " + sgn + " position.x*position.x " + sgn + " position.y*position.y)/2.; \n " ;
if ( eyes ) fmain + =
" vw *= exp(-zpos); \n " ;
else fmain + =
" position *= exp(-zpos); \n "
" zspeed = at1.z; \n "
" xspeed = length(at1.xy); \n "
" tangent = vw * exp(-zpos) * vec4(at1.xy, 0, 0) / xspeed; \n " ;
}
else if ( ! eyes ) {
fmain + =
" position = vw * vec4(0., 0., 0., 1.); \n "
" tangent = vw * at0; \n " ;
}
if ( eyes ) {
fsh + = " mediump uniform mat4 uEyeShift; \n " ;
fsh + = " mediump uniform float uAbsUnit; \n " ;
}
if ( stretch : : in ( ) ) {
if ( stretch : : mstretch ) {
fsh + = " mediump uniform mat4 uITOA; \n " ;
fsh + = " mediump uniform mat4 uATOI; \n " ;
fsh + = " mediump uniform mat4 uToOrig; \n " ;
fsh + = " mediump uniform mat4 uFromOrig; \n " ;
fsh + = " mediump mat4 toOrig; \n " ;
fsh + = " mediump mat4 fromOrig; \n " ;
fmain + =
" toOrig = uToOrig; \n "
" fromOrig = uFromOrig; \n " ;
fmain + =
" tangent = s_itranslate(toOrig * position) * toOrig * tangent; \n " ;
fmain + =
" tangent = uITOA * tangent; \n " ;
fmain + =
" tangent = fromOrig * s_translate(toOrig * position) * tangent; \n " ;
}
else {
fmain + =
" tangent = s_itranslate(position) * tangent; \n " ;
fmain + =
" tangent[2] /= " + glhr : : to_glsl ( stretch : : not_squared ( ) ) + " ; \n " ;
fmain + =
" tangent = s_translate(position) * tangent; \n " ;
}
}
}
2021-09-16 18:50:39 +00:00
void raygen : : create ( ) {
using glhr : : to_glsl ;
2021-07-09 08:09:31 +00:00
currentmap - > wall_offset ( centerover ) ; /* so raywall is not empty and deg is not zero */
2020-06-03 13:11:20 +00:00
2020-05-08 19:14:31 +00:00
deg = 0 ;
2020-05-09 16:32:16 +00:00
2021-09-16 18:50:39 +00:00
auto samples = used_sample_list ( ) ;
2020-05-09 16:32:16 +00:00
for ( int i = 0 ; i < isize ( samples ) - 1 ; i + + )
deg = max ( deg , samples [ i + 1 ] . first - samples [ i ] . first ) ;
2021-09-16 18:50:39 +00:00
if ( true ) {
asonov = hr : : asonov : : in ( ) ;
use_reflect = reflect_val & & ! nil & & ! levellines ;
many_cell_types = need_many_cell_types ( ) ;
vsh =
2019-11-14 20:21:17 +00:00
" attribute mediump vec4 aPosition; \n "
2020-12-26 20:12:34 +00:00
" uniform mediump mat4 uProjection; \n "
2019-11-14 20:21:17 +00:00
" varying mediump vec4 at; \n "
2019-10-21 22:37:57 +00:00
" void main() { \n "
2020-12-26 20:12:34 +00:00
" gl_Position = aPosition; at = uProjection * aPosition; \n "
2019-10-21 22:37:57 +00:00
" } \n " ;
2021-09-16 18:50:39 +00:00
2021-09-16 19:30:26 +00:00
if ( intra : : in ) {
irays = 0 ;
intra : : resetter ir ;
for ( int i = 0 ; i < isize ( intra : : data ) ; i + + ) {
intra : : switch_to ( i ) ;
irays + = isize ( cgi . raywall ) ;
}
}
else irays = isize ( cgi . raywall ) ;
2020-01-15 17:39:35 +00:00
string rays = its ( irays ) ;
2021-09-16 18:50:39 +00:00
fsh =
2019-11-14 20:21:17 +00:00
" varying mediump vec4 at; \n "
" uniform mediump int uLength; \n "
" uniform mediump float uIPD; \n "
" uniform mediump mat4 uStart; \n "
" uniform mediump vec2 uStartid; \n "
" uniform mediump sampler2D tConnections; \n "
" uniform mediump sampler2D tWallcolor; \n "
2020-07-07 20:16:38 +00:00
" uniform mediump sampler2D tVolumetric; \n "
2019-11-14 20:21:17 +00:00
" uniform mediump sampler2D tTexture; \n "
" uniform mediump sampler2D tTextureMap; \n "
" uniform mediump vec4 uFogColor; \n "
" uniform mediump float uLinearSightRange, uExpStart, uExpDecay; \n " ;
2021-05-17 12:21:14 +00:00
2021-09-30 14:13:47 +00:00
if ( intra : : in ) fsh + =
" uniform mediump sampler2D tPortalConnections; \n " ;
2021-05-17 12:21:14 +00:00
if ( wall_via_texture ) {
fsh + =
" uniform mediump sampler2D tWall; \n "
" uniform mediump float uInvLengthWall; \n "
" mediump vec4 getWall(mediump int x, mediump int coord) { \n "
2021-06-01 16:01:48 +00:00
" mediump vec4 result; \n "
" mediump vec4 v = texture2D(tWall, vec2((float(x)+.5) * uInvLengthWall, (float(coord)+.5) / 4.)); \n "
2021-07-07 21:33:34 +00:00
" for(int j=0; j<4; j++) result[j] = (v[j] - .5) * " + to_glsl ( ray_scale ) + " ; \n "
2021-05-17 12:21:14 +00:00
" return result; \n "
" } \n "
" mediump int getWallstart(mediump int x) { \n "
2021-06-01 16:01:48 +00:00
" mediump vec4 v = texture2D(tWall, vec2((float(x)+.5) * uInvLengthWall, 0.625)); \n "
2021-05-17 12:21:14 +00:00
" return int(v[0] / uInvLengthWall); \n "
" } \n " ;
}
else fsh + =
" uniform mediump vec4 uWallX[ " + rays + " ]; \n "
" uniform mediump vec4 uWallY[ " + rays + " ]; \n "
" uniform mediump int uWallstart[ " + its ( isize ( cgi . wallstart ) ) + " ]; \n " ;
if ( m_via_texture ) {
fsh + =
" uniform mediump sampler2D tM; \n "
" uniform mediump float uInvLengthM; \n "
" mediump mat4 getM(mediump int x) { \n "
2021-06-01 16:01:48 +00:00
" mediump mat4 result; \n "
2021-05-17 12:21:14 +00:00
" for(int i=0; i<4; i++) { \n "
2021-06-01 16:01:48 +00:00
" mediump vec4 v = texture2D(tM, vec2((float(x)+.5) * uInvLengthM, (float(i)+.5) / 4.)); \n "
2021-07-07 21:33:34 +00:00
" for(int j=0; j<4; j++) result[j][i] = (v[j] - .5) * " + to_glsl ( ray_scale ) + " ; \n "
2021-05-17 12:21:14 +00:00
" } \n "
" return result; \n "
" } \n " ;
}
else fsh + =
" uniform mediump mat4 uM[ " + its ( gms_limit ) + " ]; \n " ;
2019-10-22 08:52:17 +00:00
2020-01-15 17:39:35 +00:00
# ifdef GLES_ONLY
2021-06-01 16:02:18 +00:00
if ( ! wall_via_texture ) {
fsh + = build_getter ( " mediump vec4 " , " uWallX " , irays ) ;
fsh + = build_getter ( " mediump vec4 " , " uWallY " , irays ) ;
fsh + = build_getter ( " mediump int " , " uWallstart " , deg + 1 ) ;
}
2021-05-17 12:21:14 +00:00
if ( ! m_via_texture )
fsh + = build_getter ( " mediump mat4 " , " uM " , gms_limit ) ;
2020-01-15 17:39:35 +00:00
# endif
2021-09-16 18:50:39 +00:00
2022-12-08 18:38:06 +00:00
if ( gproduct | | intra : : in ) fsh + =
2019-11-14 20:21:17 +00:00
" uniform mediump mat4 uLP; \n " ;
2021-09-16 18:50:39 +00:00
2022-12-08 18:38:06 +00:00
if ( gproduct | | intra : : in ) fsh + =
2021-09-16 18:50:39 +00:00
" uniform mediump float uPLevel; \n " ;
if ( many_cell_types ) fsh + =
2020-05-08 19:14:31 +00:00
" uniform int uWallOffset, uSides; \n " ;
2021-09-16 18:50:39 +00:00
2020-11-14 12:53:19 +00:00
if ( many_cell_types ) {
2020-05-08 19:18:47 +00:00
fsh + = " int walloffset, sides; \n " ;
}
else {
fsh + = " const int walloffset = 0; \n "
2020-06-03 13:11:20 +00:00
" const int sides = " + its ( centerover - > type + ( WDIM = = 2 ? 2 : 0 ) ) + " ; \n " ;
2020-05-08 19:18:47 +00:00
}
2021-09-16 18:50:39 +00:00
fsh + =
2020-01-15 17:39:35 +00:00
" mediump vec2 map_texture(mediump vec4 pos, int which) { \n " ;
2021-09-16 18:50:39 +00:00
fsh + =
2021-05-17 12:21:14 +00:00
" int s = " + getWallstart ( " which " ) + " ; \n "
" int e = " + getWallstart ( " which+1 " ) + " ; \n "
2019-11-03 12:36:40 +00:00
" for(int ix=0; ix<16; ix++) { \n "
" int i = s+ix; if(i >= e) break; \n "
2021-05-17 12:21:14 +00:00
" mediump vec2 v = vec2(dot( " + getWall ( " i " , 0 ) + " , pos), dot( " + getWall ( " i " , 1 ) + " , pos)); \n "
2019-10-26 08:32:44 +00:00
" if(v.x >= 0. && v.y >= 0. && v.x + v.y <= 1.) return vec2(v.x+v.y, v.x-v.y); \n "
2019-10-22 10:07:38 +00:00
" } \n "
2019-10-26 08:32:44 +00:00
" return vec2(1, 1); \n "
2019-10-22 10:07:38 +00:00
" } \n " ;
2021-01-31 14:27:58 +00:00
2021-09-16 18:50:39 +00:00
eyes = is_eyes ( ) ;
fmain = " void main() { \n " ;
2019-10-27 00:23:20 +00:00
if ( use_reflect ) fmain + = " bool depthtoset = true; \n " ;
2021-09-16 18:50:39 +00:00
2019-10-22 08:52:17 +00:00
if ( IN_ODS ) fmain + =
2020-01-15 17:39:35 +00:00
" mediump float lambda = at[0]; \n " // -PI to PI
" mediump float phi; \n "
" mediump float eye; \n "
2019-10-21 22:37:57 +00:00
" if(at.y < 0.) { phi = at.y + PI/2.; eye = uIPD / 2.; } \n " // right
" else { phi = at.y - PI/2.; eye = -uIPD / 2.; } \n "
2021-10-03 08:55:07 +00:00
" mediump mat4 vw = uStart * xzspin(-lambda) * " + f_xpush ( ) + " (eye) * yzspin(phi); \n "
2020-01-15 17:39:35 +00:00
" mediump vec4 at0 = vec4(0., 0., 1., 0.); \n " ;
2019-10-22 08:52:17 +00:00
2020-11-01 20:20:54 +00:00
else {
fmain + =
" mediump mat4 vw = uStart; \n "
" mediump vec4 at0 = at; \n "
" gl_FragColor = vec4(0,0,0,1); \n "
" mediump float left = 1.; \n "
" at0.y = -at.y; \n "
" at0.w = 0.; \n " ;
if ( panini_alpha ) fmain + =
" mediump float hr = at0.x*at0.x; \n "
" mediump float alpha = " + to_glsl ( panini_alpha ) + " ; \n "
" mediump float A = 1. + hr; \n "
" mediump float B = -2.*hr*alpha; \n "
" mediump float C = 1. - hr*alpha*alpha; \n "
" B /= A; C /= A; \n "
" mediump float hz = B / 2. + sqrt(C + B*B/4.); \n "
" if(abs(hz) > 1e-3) { "
" at0.xyz *= hz+alpha; \n "
" at0.z = hz; \n } "
" else at0.z = 0.; \n "
2021-03-21 10:28:10 +00:00
" \n "
;
else if ( stereo_alpha ) fmain + =
" mediump float hr = at0.x*at0.x+at0.y*at0.y; \n "
" mediump float alpha = " + to_glsl ( stereo_alpha ) + " ; \n "
" mediump float A = 1. + hr; \n "
" mediump float B = -2.*hr*alpha; \n "
" mediump float C = 1. - hr*alpha*alpha; \n "
" B /= A; C /= A; \n "
" mediump float hz = B / 2. + sqrt(C + B*B/4.); \n "
" if(abs(hz) > 1e-3) { "
" at0.xyz *= hz+alpha; \n "
" at0.z = hz; \n } "
" else at0.z = 0.; \n "
" \n "
2020-11-01 20:20:54 +00:00
;
fmain + =
" at0.xyz = at0.xyz / length(at0.xyz); \n " ;
2021-01-31 14:27:58 +00:00
if ( eyes ) fmain + = " at0.xyz /= uAbsUnit; \n " ;
2020-11-01 20:20:54 +00:00
}
2019-10-22 08:52:17 +00:00
2021-01-31 14:27:58 +00:00
ld s = 1 ;
2021-02-01 10:21:51 +00:00
# if CAP_VR
2021-01-31 14:27:58 +00:00
if ( eyes ) s * = vrhr : : absolute_unit_in_meters ;
2021-02-01 10:21:51 +00:00
# endif
2021-01-31 14:27:58 +00:00
2021-10-03 16:03:14 +00:00
if ( is_stepbased ( ) | | intra : : in ) fmain + =
2021-01-31 14:27:58 +00:00
" const mediump float maxstep = " + fts ( maxstep_current ( ) * s ) + " ; \n "
" const mediump float minstep = " + fts ( minstep * s ) + " ; \n "
2021-10-09 09:11:31 +00:00
" mediump float next = maxstep; \n " ;
2019-10-23 16:52:57 +00:00
2021-10-09 09:11:31 +00:00
string fmain_prod , fmain_nprod ;
fmain + = " mediump vec4 position; \n " ;
fmain + = " mediump vec4 tangent; \n " ;
2022-12-08 18:38:06 +00:00
if ( gproduct | | intra : : in ) {
2021-09-16 19:30:26 +00:00
fmain + = " mediump float zspeed = 1.; \n " ;
fmain + = " mediump float xspeed = 1.; \n " ;
fmain + = " mediump float zpos = 0.; \n " ;
}
2021-09-16 18:50:39 +00:00
2020-11-14 12:53:19 +00:00
if ( many_cell_types ) fmain + = " walloffset = uWallOffset; sides = uSides; \n " ;
2021-09-16 18:50:39 +00:00
2021-10-09 09:11:31 +00:00
if ( intra : : in ) {
int q = isize ( intra : : data ) ;
intra : : resetter ir ;
for ( int gid2 = 0 ; gid2 < q ; gid2 + + ) {
if ( gid2 = = q - 1 )
fmain + = " { \n " ;
else {
fmain + = " if(walloffset < " + its ( intra : : data [ gid2 + 1 ] . wallindex ) + " ) { \n " ;
}
intra : : switch_to ( gid2 ) ;
emit_raystarter ( ) ;
if ( gid2 = = q - 1 )
fmain + = " } \n " ;
else
fmain + = " } else \n " ;
}
}
else emit_raystarter ( ) ;
2021-09-16 18:50:39 +00:00
fmain + =
2020-01-15 17:39:35 +00:00
" mediump float go = 0.; \n "
" mediump vec2 cid = uStartid; \n "
2019-10-25 13:18:31 +00:00
" for(int iter=0; iter< " + its ( max_iter_current ( ) ) + " ; iter++) { \n " ;
2019-10-27 00:23:20 +00:00
2021-09-16 19:30:26 +00:00
if ( intra : : in ) {
int gi = isize ( intra : : data ) ;
for ( int i = 0 ; i < gi ; i + + ) {
if ( i = = gi - 1 )
fmain + = " { \n " ;
else {
fmain + = " if(walloffset < " + its ( intra : : data [ i + 1 ] . wallindex ) + " ) { \n " ;
}
intra : : resetter ir ;
intra : : switch_to ( i ) ;
emit_iterate ( i ) ;
if ( i = = gi - 1 )
fmain + = " } \n " ;
else
fmain + = " } else \n " ;
}
}
else
emit_iterate ( - 1 ) ;
2019-10-22 08:52:17 +00:00
2019-10-28 16:33:17 +00:00
fmain + =
2019-10-26 12:11:44 +00:00
" } \n "
2019-10-27 00:23:20 +00:00
" gl_FragColor.xyz += left * uFogColor.xyz; \n " ;
2020-01-15 17:39:35 +00:00
# ifndef GLES_ONLY
2019-10-27 00:23:20 +00:00
if ( use_reflect ) fmain + =
" if(depthtoset) gl_FragDepth = 1.; \n " ;
else fmain + =
" gl_FragDepth = 1.; \n " ;
2020-01-15 17:39:35 +00:00
# endif
2019-10-27 00:23:20 +00:00
2021-09-16 18:50:39 +00:00
fmain + =
2019-10-21 22:37:57 +00:00
" } " ;
}
2021-10-03 08:55:07 +00:00
add_functions ( ) ;
2019-10-21 22:37:57 +00:00
}
2021-02-18 14:47:45 +00:00
void bind_array ( vector < array < float , 4 > > & v , GLint t , GLuint & tx , int id , int length ) {
2019-10-22 08:52:17 +00:00
if ( t = = - 1 ) println ( hlog , " bind to nothing " ) ;
2019-10-21 22:37:57 +00:00
glUniform1i ( t , id ) ;
if ( tx = = 0 ) glGenTextures ( 1 , & tx ) ;
glActiveTexture ( GL_TEXTURE0 + id ) ;
2020-10-15 14:37:43 +00:00
GLERR ( " activeTexture " ) ;
2019-10-25 22:01:12 +00:00
glBindTexture ( GL_TEXTURE_2D , tx ) ;
2020-10-15 14:37:43 +00:00
GLERR ( " bindTexture " ) ;
2019-10-21 22:37:57 +00:00
2019-10-25 22:01:12 +00:00
glTexParameteri ( GL_TEXTURE_2D , GL_TEXTURE_MIN_FILTER , GL_NEAREST ) ;
glTexParameteri ( GL_TEXTURE_2D , GL_TEXTURE_MAG_FILTER , GL_NEAREST ) ;
2020-10-15 14:37:43 +00:00
GLERR ( " texParameteri " ) ;
2019-10-21 22:37:57 +00:00
2020-10-15 14:37:43 +00:00
# ifdef GLES_ONLY
glTexImage2D ( GL_TEXTURE_2D , 0 , GL_RGBA , length , isize ( v ) / length , 0 , GL_RGBA , GL_FLOAT , & v [ 0 ] ) ;
# else
2019-10-26 15:03:29 +00:00
glTexImage2D ( GL_TEXTURE_2D , 0 , 0x8814 /* GL_RGBA32F */ , length , isize ( v ) / length , 0 , GL_RGBA , GL_FLOAT , & v [ 0 ] ) ;
2020-10-15 14:37:43 +00:00
# endif
2019-10-21 22:37:57 +00:00
GLERR ( " bind_array " ) ;
}
2019-10-25 22:01:12 +00:00
void uniform2 ( GLint id , array < float , 2 > fl ) {
glUniform2f ( id , fl [ 0 ] , fl [ 1 ] ) ;
2019-10-25 13:05:17 +00:00
}
2019-12-06 13:14:27 +00:00
color_t color_out_of_range = 0x0F0800FF ;
2019-10-25 22:17:58 +00:00
2022-02-26 09:40:39 +00:00
EX transmatrix get_ms ( cell * c , int a , bool mirror ) {
2020-07-03 12:54:50 +00:00
int z = a ? 1 : - 1 ;
if ( c - > type = = 3 ) {
hyperpoint h =
project_on_triangle (
hybrid : : get_corner ( c , a , 0 , z ) ,
hybrid : : get_corner ( c , a , 1 , z ) ,
hybrid : : get_corner ( c , a , 2 , z )
) ;
transmatrix T = rspintox ( h ) ;
if ( mirror ) T = T * MirrorX ;
return T * xpush ( - 2 * hdist0 ( h ) ) * spintox ( h ) ;
}
else {
hyperpoint h = Hypc ;
for ( int a = 0 ; a < c - > type ; a + + ) {
hyperpoint corner = hybrid : : get_corner ( c , a , 0 , z ) ;
h + = corner ;
}
h = normalize ( h ) ;
ld d = hdist0 ( h ) ;
2022-12-15 20:17:08 +00:00
if ( moved_center ( ) ) d - = 1 ;
2020-07-03 12:54:50 +00:00
if ( h [ 2 ] > 0 ) d = - d ;
2022-12-15 20:17:08 +00:00
if ( mirror ) return MirrorZ * lzpush ( 2 * d ) ;
return lzpush ( 2 * d ) ;
2020-07-03 12:54:50 +00:00
}
}
2020-11-14 12:53:19 +00:00
int nesting ;
2022-02-26 09:40:39 +00:00
EX transmatrix mirrorize ( transmatrix T ) {
2021-07-09 18:12:35 +00:00
T = inverse ( T ) ;
hyperpoint h = tC0 ( T ) ;
ld d = hdist0 ( h ) ;
return rspintox ( h ) * xpush ( d / 2 ) * MirrorX * xpush ( - d / 2 ) * spintox ( h ) ;
}
2021-09-16 18:53:05 +00:00
transmatrix protect_prod ( transmatrix T ) {
if ( MDIM = = 3 )
for ( int i = 0 ; i < 4 ; i + + ) T [ i ] [ 3 ] = T [ 3 ] [ i ] = i = = 3 ;
return T ;
}
2021-02-18 14:47:45 +00:00
struct raycast_map {
2021-03-09 13:09:58 +00:00
int saved_frameid ;
2022-10-26 22:02:35 +00:00
int saved_map_version ;
2021-02-18 14:47:45 +00:00
vector < cell * > lst ;
map < cell * , int > ids ;
vector < transmatrix > ms ;
2021-09-16 18:50:39 +00:00
int length , per_row , rows , mirror_shift , deg ;
2021-02-18 14:47:45 +00:00
2021-09-30 14:13:47 +00:00
vector < array < float , 4 > > connections , wallcolor , texturemap , volumetric , portal_connections ;
2021-02-18 14:47:45 +00:00
void apply_shape ( ) {
length = 4096 ;
2021-09-16 18:50:39 +00:00
deg = our_raygen . deg ;
per_row = length / deg ;
2021-02-18 14:47:45 +00:00
rows = next_p2 ( ( isize ( lst ) + per_row - 1 ) / per_row ) ;
int q = length * rows ;
connections . resize ( q ) ;
2021-09-30 14:13:47 +00:00
portal_connections . resize ( q ) ;
2021-02-18 14:47:45 +00:00
wallcolor . resize ( q ) ;
texturemap . resize ( q ) ;
volumetric . resize ( q ) ;
}
2021-07-09 18:12:35 +00:00
2021-02-18 14:47:45 +00:00
void generate_initial_ms ( cell * cs ) {
2021-09-16 18:50:39 +00:00
auto sa = used_sample_list ( ) ;
2021-02-18 14:47:45 +00:00
ms . clear ( ) ;
ms . resize ( sa . back ( ) . first , Id ) ;
2021-09-16 19:30:26 +00:00
intra : : resetter ir ;
2021-02-18 14:47:45 +00:00
for ( auto & p : sa ) {
int id = p . first ;
cell * c = p . second ;
if ( ! c ) continue ;
2021-10-03 16:06:39 +00:00
intra : : may_switch_to ( c ) ;
2021-02-18 14:47:45 +00:00
for ( int j = 0 ; j < c - > type ; j + + )
2021-09-16 18:53:05 +00:00
ms [ id + j ] = protect_prod ( currentmap - > ray_iadj ( c , j ) ) ;
2021-02-18 14:47:45 +00:00
if ( WDIM = = 2 ) for ( int a : { 0 , 1 } ) {
ms [ id + c - > type + a ] = get_ms ( c , a , false ) ;
}
}
// println(hlog, ms);
2021-07-09 18:12:35 +00:00
mirror_shift = isize ( ms ) ;
2021-02-18 14:47:45 +00:00
if ( ! sol & & ! nil & & ( reflect_val | | reg3 : : ultra_mirror_in ( ) ) ) {
2021-07-09 18:12:35 +00:00
ms . resize ( mirror_shift * 2 ) ;
2021-02-18 14:47:45 +00:00
2021-07-09 18:12:35 +00:00
for ( auto & p : sa ) {
int id = p . first ;
cell * c = p . second ;
if ( ! c ) continue ;
2021-09-16 19:30:26 +00:00
intra : : may_switch_to ( c ) ;
2021-07-09 18:12:35 +00:00
for ( int j = 0 ; j < c - > type ; j + + )
2021-09-16 18:53:05 +00:00
ms [ mirror_shift + id + j ] = protect_prod ( mirrorize ( ms [ id + j ] ) ) ;
2021-07-09 18:12:35 +00:00
if ( WDIM = = 2 ) for ( int a : { 0 , 1 } ) {
ms [ mirror_shift + id + c - > type + a ] = get_ms ( c , a , true ) ;
2021-02-18 14:47:45 +00:00
}
2021-07-09 18:12:35 +00:00
}
2021-02-18 14:47:45 +00:00
if ( reg3 : : ultra_mirror_in ( ) ) {
for ( auto v : cgi . ultra_mirrors )
ms . push_back ( v ) ;
}
}
2021-09-16 19:30:26 +00:00
for ( auto p : sa ) {
cell * c = p . second ;
if ( ! c ) continue ;
intra : : may_switch_to ( c ) ;
int id = p . first ;
2022-12-08 18:38:06 +00:00
if ( gproduct ) {
2021-09-16 19:30:26 +00:00
ms [ id + c - > type - 2 ] = Id ;
ms [ id + c - > type - 1 ] = Id ;
2021-02-18 14:47:45 +00:00
}
}
}
void generate_cell_listing ( cell * cs ) {
manual_celllister cl ;
cl . add ( cs ) ;
bool optimize = ! isWall3 ( cs ) ;
2021-09-16 19:30:26 +00:00
intra : : resetter ir ;
2021-02-18 14:47:45 +00:00
// vector<int> legaldir = { -1 };
for ( int i = 0 ; i < isize ( cl . lst ) ; i + + ) {
cell * c = cl . lst [ i ] ;
2021-09-16 19:30:26 +00:00
intra : : may_switch_to ( c ) ;
2021-02-18 14:47:45 +00:00
if ( racing : : on & & i > 0 & & c - > wall = = waBarrier ) continue ;
if ( optimize & & isWall3 ( c ) ) continue ;
forCellIdCM ( c2 , d , c ) {
// if(reflect_val == 0 && !((1<<d) & legaldir[i])) continue;
2021-09-16 19:30:26 +00:00
2021-10-09 13:45:59 +00:00
if ( intra : : in & & ! intra : : intra_id . count ( c2 ) ) intra : : intra_id [ c2 ] = intra : : current ;
2021-09-16 19:30:26 +00:00
if ( intra : : in ) {
cellwalker cw ( c , d ) ;
auto p = at_or_null ( intra : : connections , cw ) ;
2021-10-09 13:45:59 +00:00
if ( p ) {
cell * c3 = p - > tcw . at ;
if ( rays_generate & & c3 - > mpdist > 7 ) { intra : : may_switch_to ( c3 ) ; setdist ( c3 , 7 , c ) ; intra : : may_switch_to ( c2 ) ; }
cl . add ( c3 ) ;
}
2021-09-16 19:30:26 +00:00
}
2021-02-18 14:47:45 +00:00
if ( rays_generate ) setdist ( c2 , 7 , c ) ;
/* if(!cl.listed(c2))
legaldir . push_back ( legaldir [ i ] & ~ ( 1 < < ( ( d + 3 ) % 6 ) ) ) ; */
cl . add ( c2 ) ;
if ( isize ( cl . lst ) > = max_cells ) goto finish ;
}
}
finish :
lst = cl . lst ;
ids . clear ( ) ;
for ( int i = 0 ; i < isize ( lst ) ; i + + ) ids [ lst [ i ] ] = i ;
}
array < float , 2 > enc ( int i , int a ) {
array < float , 2 > res ;
res [ 0 ] = ( ( i % per_row ) * deg + a + .5 ) / length ;
res [ 1 ] = ( ( i / per_row ) + .5 ) / rows ;
return res ;
}
void generate_connections ( cell * c , int id ) {
2021-09-16 19:30:26 +00:00
intra : : may_switch_to ( c ) ;
2021-02-18 14:47:45 +00:00
auto & vmap = volumetric : : vmap ;
if ( volumetric : : on ) {
celldrawer dd ;
dd . c = c ;
dd . setcolors ( ) ;
int u = ( id / per_row * length ) + ( id % per_row * deg ) ;
color_t vcolor ;
if ( vmap . count ( c ) )
vcolor = vmap [ c ] ;
else
vcolor = ( backcolor < < 8 ) ;
volumetric [ u ] = glhr : : acolor ( vcolor ) ;
}
2021-09-16 19:30:26 +00:00
forCellIdEx ( c1_real , i , c ) {
cell * c1 = c1_real ;
const intra : : connection_data * p = nullptr ;
if ( intra : : in ) {
cellwalker cw ( c , i ) ;
p = at_or_null ( intra : : connections , cw ) ;
if ( p ) c1 = p - > tcw . at ;
}
2021-02-18 14:47:45 +00:00
int u = ( id / per_row * length ) + ( id % per_row * deg ) + i ;
if ( ! ids . count ( c1 ) ) {
wallcolor [ u ] = glhr : : acolor ( color_out_of_range | 0xFF ) ;
texturemap [ u ] = glhr : : makevertex ( 0.1 , 0 , 0 ) ;
continue ;
}
auto code = enc ( ids [ c1 ] , 0 ) ;
connections [ u ] [ 0 ] = code [ 0 ] ;
connections [ u ] [ 1 ] = code [ 1 ] ;
2021-09-30 14:13:47 +00:00
portal_connections [ u ] [ 0 ] = 0 ;
2021-02-18 14:47:45 +00:00
if ( isWall3 ( c1 ) ) {
celldrawer dd ;
dd . c = c1 ;
dd . setcolors ( ) ;
shiftmatrix Vf ;
dd . set_land_floor ( Vf ) ;
color_t wcol = darkena ( dd . wcol , 0 , 0xFF ) ;
int dv = get_darkval ( c1 , c - > c . spin ( i ) ) ;
float p = 1 - dv / 16. ;
wallcolor [ u ] = glhr : : acolor ( wcol ) ;
for ( int a : { 0 , 1 , 2 } ) wallcolor [ u ] [ a ] * = p ;
if ( qfi . fshape ) {
texturemap [ u ] = floor_texture_map [ qfi . fshape - > id ] ;
}
else
texturemap [ u ] = glhr : : makevertex ( 0.1 , 0 , 0 ) ;
}
else {
color_t col = transcolor ( c , c1 , winf [ c - > wall ] . color ) | transcolor ( c1 , c , winf [ c1 - > wall ] . color ) ;
if ( col = = 0 )
wallcolor [ u ] = glhr : : acolor ( 0 ) ;
else {
int dv = get_darkval ( c1 , c - > c . spin ( i ) ) ;
float p = 1 - dv / 16. ;
wallcolor [ u ] = glhr : : acolor ( col ) ;
for ( int a : { 0 , 1 , 2 } ) wallcolor [ u ] [ a ] * = p ;
texturemap [ u ] = glhr : : makevertex ( 0.001 , 0 , 0 ) ;
}
}
2021-09-16 19:30:26 +00:00
int wo = intra : : full_wall_offset ( c ) ;
2021-09-16 18:53:05 +00:00
if ( wo > = our_raygen . irays ) {
println ( hlog , " wo= " , wo , " irays = " , our_raygen . irays ) ;
2021-02-18 14:47:45 +00:00
reset_raycaster ( ) ;
return ;
}
2021-09-16 19:30:26 +00:00
if ( p ) {
int k = isize ( ms ) ;
auto bak = geometry ;
ms . push_back ( p - > T ) ;
geometry = bak ;
ms . push_back ( p - > id1 . T ) ;
ms . push_back ( p - > id2 . iT ) ;
2021-02-18 14:47:45 +00:00
connections [ u ] [ 2 ] = ( k + .5 ) / 1024. ;
2021-09-30 14:13:47 +00:00
portal_connections [ u ] [ 0 ] = p - > id1 . kind / 16. + .5 ;
portal_connections [ u ] [ 1 ] = p - > id1 . d / 16 + .5 ;
portal_connections [ u ] [ 2 ] = p - > id2 . kind / 16. + .5 ;
portal_connections [ u ] [ 3 ] = p - > id2 . d / 16 + .5 ;
2021-02-18 14:47:45 +00:00
}
2021-09-16 19:30:26 +00:00
else {
transmatrix T = currentmap - > iadj ( c , i ) * inverse ( ms [ wo + i ] ) ;
if ( in_e2xe ( ) & & i > = c - > type - 2 )
T = Id ;
T = protect_prod ( T ) ;
for ( int k = 0 ; k < = isize ( ms ) ; k + + ) {
if ( k < isize ( ms ) & & ! eqmatrix ( ms [ k ] , T , 1e-5 ) ) continue ;
if ( k = = isize ( ms ) ) ms . push_back ( T ) ;
connections [ u ] [ 2 ] = ( k + .5 ) / 1024. ;
break ;
}
}
intra : : resetter ir ;
intra : : may_switch_to ( c1 ) ;
int wo1 = intra : : full_wall_offset ( c1 ) ;
2021-07-09 08:09:31 +00:00
if ( wo1 > = max_wall_offset )
println ( hlog , " error: wall_offset " , wo1 , " exceeds " , max_wall_offset ) ;
2021-07-07 21:33:34 +00:00
if ( c1 - > type > = max_celltype )
println ( hlog , " error: type " + its ( c1 - > type ) + " exceeds " , max_celltype ) ;
2021-07-09 08:09:31 +00:00
connections [ u ] [ 3 ] = ( wo1 * 1. / max_wall_offset ) + ( c1 - > type + ( WDIM = = 2 ? 2 : 0 ) + .5 ) * 1. / max_wall_offset / max_celltype ;
2021-02-18 14:47:45 +00:00
}
if ( WDIM = = 2 ) for ( int a : { 0 , 1 } ) {
celldrawer dd ;
dd . c = c ;
dd . setcolors ( ) ;
shiftmatrix Vf ;
dd . set_land_floor ( Vf ) ;
int u = ( id / per_row * length ) + ( id % per_row * deg ) + c - > type + a ;
wallcolor [ u ] = glhr : : acolor ( darkena ( dd . fcol , 0 , 0xFF ) ) ;
if ( qfi . fshape )
texturemap [ u ] = floor_texture_map [ qfi . fshape - > id ] ;
else
texturemap [ u ] = glhr : : makevertex ( 0.1 , 0 , 0 ) ;
}
}
void generate_connections ( ) {
int id = 0 ;
2021-09-16 19:30:26 +00:00
intra : : resetter ir ;
2021-08-06 13:38:54 +00:00
for ( cell * c : lst ) if ( ! reset_rmap )
2021-02-18 14:47:45 +00:00
generate_connections ( c , id + + ) ;
}
bool gms_exceeded ( ) {
2021-06-01 16:02:58 +00:00
if ( m_via_texture ) return false ;
2021-02-18 14:47:45 +00:00
return isize ( ms ) > gms_array_size ;
}
void assign_uniforms ( raycaster * o ) {
2021-05-17 12:21:29 +00:00
if ( ! o ) return ;
2021-02-18 14:47:45 +00:00
glUniform1i ( o - > uLength , length ) ;
GLERR ( " uniform mediump length " ) ;
2021-05-17 12:21:14 +00:00
if ( m_via_texture ) {
int mlength = next_p2 ( isize ( ms ) ) ;
vector < array < float , 4 > > m_map ;
m_map . resize ( 4 * mlength ) ;
for ( int i = 0 ; i < isize ( ms ) ; i + + )
for ( int a = 0 ; a < 4 ; a + + )
2021-07-09 18:33:33 +00:00
for ( int b = 0 ; b < 4 ; b + + ) {
2021-09-16 18:53:05 +00:00
m_map [ i + a * mlength ] [ b ] = ms [ i ] [ a ] [ b ] / ray_scale + .5 ;
2021-07-09 18:33:33 +00:00
}
2021-05-17 12:21:14 +00:00
bind_array ( m_map , o - > tM , txM , 7 , mlength ) ;
glUniform1f ( o - > uInvLengthM , 1. / mlength ) ;
}
else {
vector < glhr : : glmatrix > gms ;
for ( auto & m : ms ) gms . push_back ( glhr : : tmtogl_transpose3 ( m ) ) ;
glUniformMatrix4fv ( o - > uM , isize ( gms ) , 0 , gms [ 0 ] . as_array ( ) ) ;
}
2021-02-18 14:47:45 +00:00
bind_array ( wallcolor , o - > tWallcolor , txWallcolor , 4 , length ) ;
bind_array ( connections , o - > tConnections , txConnections , 3 , length ) ;
bind_array ( texturemap , o - > tTextureMap , txTextureMap , 5 , length ) ;
if ( volumetric : : on ) bind_array ( volumetric , o - > tVolumetric , txVolumetric , 6 , length ) ;
2021-10-07 16:23:57 +00:00
if ( o - > tPortalConnections ! = - 1 )
bind_array ( portal_connections , o - > tPortalConnections , txPortalConnections , 1 , length ) ;
2021-07-09 18:12:35 +00:00
if ( o - > uMirrorShift ! = - 1 ) {
glUniform1i ( o - > uMirrorShift , mirror_shift ) ;
}
2021-02-18 14:47:45 +00:00
}
void create_all ( cell * cs ) {
2021-03-09 13:09:58 +00:00
saved_frameid = frameid ;
2022-10-26 22:02:35 +00:00
saved_map_version = mapeditor : : map_version ;
2021-02-18 14:47:45 +00:00
generate_initial_ms ( cs ) ;
generate_cell_listing ( cs ) ;
apply_shape ( ) ;
generate_connections ( ) ;
}
bool need_to_create ( cell * cs ) {
2021-03-09 13:09:58 +00:00
if ( ! fixed_map & & frameid ! = saved_frameid ) return true ;
2022-10-26 22:02:35 +00:00
if ( saved_map_version ! = mapeditor : : map_version ) return true ;
2021-02-18 14:47:45 +00:00
return ! ids . count ( cs ) ;
}
} ;
unique_ptr < raycast_map > rmap ;
2021-08-06 13:38:54 +00:00
EX bool reset_rmap = false ;
2021-02-18 14:47:45 +00:00
2021-12-11 23:30:44 +00:00
EX cell * rmap_get_by_id ( int id ) {
return rmap - > lst [ id ] ;
}
EX int rmap_get_id_of ( cell * c ) {
return rmap - > ids [ c ] ;
}
2021-04-07 00:32:38 +00:00
EX void reset_raycaster ( ) {
2021-05-12 18:19:45 +00:00
our_raycaster = nullptr ;
reset_rmap = true ;
2021-04-07 00:32:38 +00:00
rots : : saved_matrices_ray = { } ;
}
2021-02-18 14:47:45 +00:00
2021-04-30 17:45:34 +00:00
EX void reset_raycaster_map ( ) {
rmap = nullptr ;
}
2021-09-16 18:50:39 +00:00
EX void load_walls ( vector < glvertex > & wallx , vector < glvertex > & wally , vector < GLint > & wallstart ) {
int q = 0 ;
if ( isize ( wallx ) ) {
q = isize ( wallx ) ;
wallstart . pop_back ( ) ;
}
for ( auto i : cgi . wallstart ) wallstart . push_back ( q + i ) ;
dynamicval < eGeometry > g ( geometry , gCubeTiling ) ;
for ( auto & m : cgi . raywall ) {
wallx . push_back ( glhr : : pointtogl ( m [ 0 ] ) ) ;
wally . push_back ( glhr : : pointtogl ( m [ 1 ] ) ) ;
}
}
2019-10-25 12:50:48 +00:00
EX void cast ( ) {
2020-11-14 12:53:19 +00:00
// may call itself recursively in case of bugs -- just in case...
dynamicval < int > dn ( nesting , nesting + 1 ) ;
if ( nesting > 10 ) return ;
2021-09-16 18:50:39 +00:00
if ( isize ( cgi . raywall ) > our_raygen . irays ) reset_raycaster ( ) ;
2020-11-14 12:53:19 +00:00
2019-10-21 22:37:57 +00:00
enable_raycaster ( ) ;
2020-11-14 12:53:19 +00:00
auto & o = our_raycaster ;
if ( need_many_cell_types ( ) & & o - > uWallOffset = = - 1 ) {
reset_raycaster ( ) ;
cast ( ) ;
return ;
}
2019-10-23 21:10:50 +00:00
2019-10-25 12:50:48 +00:00
if ( comparison_mode )
2019-10-23 21:10:50 +00:00
glColorMask ( GL_TRUE , GL_FALSE , GL_FALSE , GL_TRUE ) ;
2019-10-21 22:37:57 +00:00
vector < glvertex > screen = {
glhr : : makevertex ( - 1 , - 1 , 1 ) ,
glhr : : makevertex ( - 1 , + 1 , 1 ) ,
glhr : : makevertex ( + 1 , - 1 , 1 ) ,
glhr : : makevertex ( - 1 , + 1 , 1 ) ,
glhr : : makevertex ( + 1 , - 1 , 1 ) ,
glhr : : makevertex ( + 1 , + 1 , 1 )
} ;
2019-11-16 02:04:01 +00:00
ld d = current_display - > eyewidth ( ) ;
if ( vid . stereo_mode = = sLR ) d = 2 * d - 1 ;
else d = - d ;
2019-10-21 22:37:57 +00:00
auto & cd = current_display ;
2019-11-16 02:04:01 +00:00
cd - > set_viewport ( global_projection ) ;
cd - > set_mask ( global_projection ) ;
2020-12-26 20:12:34 +00:00
2020-12-27 16:37:39 +00:00
# if CAP_VR
2021-01-31 14:27:58 +00:00
if ( o - > uEyeShift ! = - 1 ) {
2022-02-28 16:34:00 +00:00
dynamicval < eGeometry > g ( geometry , gCubeTiling ) ;
2021-02-01 21:51:24 +00:00
transmatrix T = vrhr : : eyeshift ;
if ( nonisotropic )
T = inverse ( NLP ) * T ;
2022-02-28 16:34:00 +00:00
glUniformMatrix4fv ( o - > uEyeShift , 1 , 0 , glhr : : tmtogl_transpose ( T ) . as_array ( ) ) ;
2021-01-31 14:27:58 +00:00
glUniform1f ( o - > uAbsUnit , vrhr : : absolute_unit_in_meters ) ;
}
2020-12-30 13:20:30 +00:00
if ( vrhr : : rendering_eye ( ) ) {
2022-02-28 16:34:00 +00:00
dynamicval < eGeometry > g ( geometry , gCubeTiling ) ;
glUniformMatrix4fv ( o - > uProjection , 1 , 0 , glhr : : tmtogl_transpose ( vrhr : : eyeproj ) . as_array ( ) ) ;
2020-12-26 20:13:32 +00:00
}
2020-12-27 16:37:39 +00:00
# else
2021-01-31 14:28:14 +00:00
if ( 0 ) ;
2020-12-27 16:37:39 +00:00
# endif
2020-12-26 20:13:32 +00:00
else {
2022-02-28 16:34:00 +00:00
dynamicval < eGeometry > g ( geometry , gCubeTiling ) ;
2020-12-26 20:13:32 +00:00
transmatrix proj = Id ;
proj = eupush ( - global_projection * d , 0 ) * proj ;
proj = euscale ( cd - > tanfov / ( vid . stereo_mode = = sLR ? 2 : 1 ) , cd - > tanfov * cd - > ysize / cd - > xsize ) * proj ;
proj = eupush ( - ( ( cd - > xcenter - cd - > xtop ) * 2. / cd - > xsize - 1 ) , - ( ( cd - > ycenter - cd - > ytop ) * 2. / cd - > ysize - 1 ) ) * proj ;
2022-02-28 16:34:00 +00:00
glUniformMatrix4fv ( o - > uProjection , 1 , 0 , glhr : : tmtogl_transpose ( proj ) . as_array ( ) ) ;
2020-12-26 20:13:32 +00:00
}
2019-10-21 22:37:57 +00:00
2020-03-29 15:35:42 +00:00
if ( ! callhandlers ( false , hooks_rayset , o ) ) {
2019-10-28 16:33:17 +00:00
2019-11-13 23:26:50 +00:00
cell * cs = centerover ;
2019-10-28 16:33:17 +00:00
2020-07-27 16:49:04 +00:00
transmatrix T = cview ( ) . T ;
2019-11-16 02:04:01 +00:00
if ( global_projection )
T = xpush ( vid . ipd * global_projection / 2 ) * T ;
2019-11-14 16:20:55 +00:00
if ( nonisotropic ) T = NLP * T ;
2019-10-28 16:33:17 +00:00
T = inverse ( T ) ;
2019-11-14 18:33:55 +00:00
virtualRebase ( cs , T ) ;
2019-10-28 16:33:17 +00:00
2020-07-03 12:55:16 +00:00
int ray_fixes = 0 ;
2020-08-22 22:11:37 +00:00
transmatrix msm = stretch : : mstretch_matrix ;
2022-12-15 20:17:08 +00:00
hyperpoint TC0 = tile_center ( ) ;
2020-07-03 12:55:16 +00:00
back :
for ( int a = 0 ; a < cs - > type ; a + + )
2022-12-15 20:17:08 +00:00
if ( hdist ( currentmap - > ray_iadj ( cs , a ) * T * C0 , TC0 ) < hdist ( T * C0 , TC0 ) ) {
2020-07-03 12:55:16 +00:00
T = currentmap - > iadj ( cs , a ) * T ;
2020-08-22 22:11:37 +00:00
if ( o - > uToOrig ! = - 1 ) {
transmatrix HT = currentmap - > adj ( cs , a ) ;
HT = stretch : : itranslate ( tC0 ( HT ) ) * HT ;
msm = HT * msm ;
}
2020-07-03 12:55:16 +00:00
cs = cs - > move ( a ) ;
ray_fixes + + ;
2021-04-07 00:33:28 +00:00
if ( ray_fixes > 100 ) {
println ( hlog , " major ray error " ) ;
return ;
}
2020-07-03 12:55:16 +00:00
goto back ;
}
2021-09-16 18:50:39 +00:00
if ( ray_fixes ) println ( hlog , " ray error x " , ray_fixes , " centerover = " , centerover , " -> " , cs ) ;
2020-07-03 12:55:16 +00:00
2019-10-28 16:33:17 +00:00
glUniformMatrix4fv ( o - > uStart , 1 , 0 , glhr : : tmtogl_transpose3 ( T ) . as_array ( ) ) ;
2019-11-14 16:20:55 +00:00
if ( o - > uLP ! = - 1 ) glUniformMatrix4fv ( o - > uLP , 1 , 0 , glhr : : tmtogl_transpose3 ( inverse ( NLP ) ) . as_array ( ) ) ;
2019-11-14 20:21:17 +00:00
GLERR ( " uniform mediump startid " ) ;
2019-10-21 22:37:57 +00:00
glUniform1f ( o - > uIPD , vid . ipd ) ;
2019-11-14 20:21:17 +00:00
GLERR ( " uniform mediump IPD " ) ;
2020-05-08 19:14:31 +00:00
2020-08-08 14:08:51 +00:00
if ( o - > uITOA ! = - 1 ) {
glUniformMatrix4fv ( o - > uITOA , 1 , 0 , glhr : : tmtogl_transpose3 ( stretch : : m_itoa ) . as_array ( ) ) ;
glUniformMatrix4fv ( o - > uATOI , 1 , 0 , glhr : : tmtogl_transpose3 ( stretch : : m_atoi ) . as_array ( ) ) ;
}
2020-08-22 22:10:59 +00:00
if ( o - > uToOrig ! = - 1 ) {
glUniformMatrix4fv ( o - > uToOrig , 1 , 0 , glhr : : tmtogl_transpose3 ( msm ) . as_array ( ) ) ;
glUniformMatrix4fv ( o - > uFromOrig , 1 , 0 , glhr : : tmtogl_transpose3 ( inverse ( msm ) ) . as_array ( ) ) ;
}
2020-08-08 14:08:51 +00:00
2020-05-08 19:14:31 +00:00
if ( o - > uWallOffset ! = - 1 ) {
2021-09-16 19:30:26 +00:00
glUniform1i ( o - > uWallOffset , intra : : full_wall_offset ( cs ) ) ;
2020-06-03 13:44:34 +00:00
glUniform1i ( o - > uSides , cs - > type + ( WDIM = = 2 ? 2 : 0 ) ) ;
2020-05-08 19:14:31 +00:00
}
2020-02-07 18:13:18 +00:00
2019-10-22 10:07:38 +00:00
vector < glvertex > wallx , wally ;
2021-09-16 18:50:39 +00:00
vector < GLint > wallstart ;
2021-09-16 19:30:26 +00:00
if ( intra : : in ) {
intra : : resetter ir ;
for ( int i = 0 ; i < isize ( intra : : data ) ; i + + ) {
intra : : switch_to ( i ) ;
load_walls ( wallx , wally , wallstart ) ;
}
}
else
load_walls ( wallx , wally , wallstart ) ;
2021-09-16 18:50:39 +00:00
2021-05-17 12:21:14 +00:00
if ( wall_via_texture ) {
int wlength = next_p2 ( isize ( wallx ) ) ;
vector < array < float , 4 > > w_map ;
w_map . resize ( 4 * wlength ) ;
ld minval = 9 , maxval = - 9 ;
for ( int i = 0 ; i < isize ( wallx ) ; i + + ) {
for ( int a = 0 ; a < 4 ; a + + ) {
2021-07-07 21:33:34 +00:00
w_map [ i ] [ a ] = wallx [ i ] [ a ] / ray_scale + .5 ;
w_map [ i + wlength ] [ a ] = wally [ i ] [ a ] / ray_scale + .5 ;
2021-05-17 12:21:14 +00:00
minval = min < ld > ( minval , w_map [ i ] [ a ] ) ;
minval = min < ld > ( minval , w_map [ i + wlength ] [ a ] ) ;
maxval = max < ld > ( maxval , w_map [ i ] [ a ] ) ;
maxval = max < ld > ( maxval , w_map [ i + wlength ] [ a ] ) ;
}
}
// println(hlog, "wallrange = ", tie(minval, maxval), " wallx = ", isize(wallx), " wallstart = ", isize(cgi.wallstart));
2021-09-16 18:50:39 +00:00
for ( int i = 0 ; i < isize ( wallstart ) ; i + + )
w_map [ i + 2 * wlength ] [ 0 ] = ( wallstart [ i ] + .5 ) / wlength ;
2021-05-17 12:21:14 +00:00
bind_array ( w_map , o - > tWall , txWall , 8 , wlength ) ;
glUniform1f ( o - > uInvLengthWall , 1. / wlength ) ;
}
else {
glUniform1iv ( o - > uWallstart , isize ( wallstart ) , & wallstart [ 0 ] ) ;
glUniform4fv ( o - > uWallX , isize ( wallx ) , & wallx [ 0 ] [ 0 ] ) ;
glUniform4fv ( o - > uWallY , isize ( wally ) , & wally [ 0 ] [ 0 ] ) ;
}
2019-10-22 10:07:38 +00:00
2019-11-03 12:36:06 +00:00
if ( o - > uLevelLines ! = - 1 )
glUniform1f ( o - > uLevelLines , levellines ) ;
2019-10-22 20:42:48 +00:00
if ( o - > uBinaryWidth ! = - 1 )
2019-11-02 10:51:44 +00:00
glUniform1f ( o - > uBinaryWidth , vid . binary_width / 2 * ( nih ? 1 : log ( 2 ) ) ) ;
2021-03-30 09:41:52 +00:00
# if CAP_SOLV
2019-11-08 14:01:03 +00:00
if ( o - > uStraighten ! = - 1 ) {
2019-11-08 14:35:23 +00:00
glUniformMatrix4fv ( o - > uStraighten , 1 , 0 , glhr : : tmtogl_transpose ( asonov : : straighten ) . as_array ( ) ) ;
2019-11-08 14:01:03 +00:00
}
2019-11-09 11:12:16 +00:00
if ( o - > uReflectX ! = - 1 ) {
2022-11-12 21:38:45 +00:00
auto h = glhr : : pointtogl ( tangent_length ( spin90 ( ) * asonov : : ty , 2 ) ) ;
2019-11-09 11:12:16 +00:00
glUniform4fv ( o - > uReflectX , 1 , & h [ 0 ] ) ;
2022-11-12 21:38:45 +00:00
h = glhr : : pointtogl ( tangent_length ( spin90 ( ) * asonov : : tx , 2 ) ) ;
2019-11-09 11:12:16 +00:00
glUniform4fv ( o - > uReflectY , 1 , & h [ 0 ] ) ;
}
2021-03-30 09:41:52 +00:00
# endif
2019-10-28 16:33:17 +00:00
if ( o - > uPLevel ! = - 1 )
glUniform1f ( o - > uPLevel , cgi . plevel / 2 ) ;
2020-10-15 14:33:52 +00:00
# if CAP_BT
2019-11-02 09:40:22 +00:00
if ( o - > uBLevel ! = - 1 )
2019-12-14 11:05:01 +00:00
glUniform1f ( o - > uBLevel , log ( bt : : expansion ( ) ) / 2 ) ;
2020-10-15 14:33:52 +00:00
# endif
2019-10-28 16:33:17 +00:00
2020-07-07 20:16:38 +00:00
if ( o - > uLinearSightRange ! = - 1 )
glUniform1f ( o - > uLinearSightRange , sightranges [ geometry ] ) ;
2019-10-25 13:05:17 +00:00
glUniform1f ( o - > uExpDecay , exp_decay_current ( ) ) ;
2020-07-07 20:16:38 +00:00
2019-10-25 13:05:17 +00:00
glUniform1f ( o - > uExpStart , exp_start ) ;
2019-10-26 13:33:57 +00:00
auto cols = glhr : : acolor ( darkena ( backcolor , 0 , 0xFF ) ) ;
2020-07-08 10:43:30 +00:00
if ( o - > uFogColor ! = - 1 )
glUniform4f ( o - > uFogColor , cols [ 0 ] , cols [ 1 ] , cols [ 2 ] , cols [ 3 ] ) ;
2021-02-18 14:47:45 +00:00
2021-05-12 18:19:45 +00:00
if ( reset_rmap ) rmap = nullptr , reset_rmap = false ;
2021-02-18 14:47:45 +00:00
if ( ! rmap ) rmap = ( unique_ptr < raycast_map > ) new raycast_map ;
2020-03-29 15:35:42 +00:00
2021-02-18 14:47:45 +00:00
if ( rmap - > need_to_create ( cs ) ) {
rmap - > create_all ( cs ) ;
2021-08-06 13:38:54 +00:00
if ( reset_rmap ) {
reset_raycaster ( ) ;
cast ( ) ;
return ;
}
2021-02-18 14:47:45 +00:00
if ( rmap - > gms_exceeded ( ) ) {
2021-06-01 16:02:58 +00:00
if ( isize ( rmap - > ms ) > gms_limit | | can_via_texture ) {
m_via_texture = true ;
wall_via_texture = true ;
println ( hlog , " enabling m_via_texture " ) ;
reset_raycaster ( ) ;
cast ( ) ;
}
else {
gms_array_size = isize ( rmap - > ms ) ;
println ( hlog , " changing gms_array_size to " , gms_array_size ) ;
reset_raycaster ( ) ;
cast ( ) ;
}
2021-02-18 14:47:45 +00:00
return ;
}
rmap - > assign_uniforms ( & * o ) ;
}
2021-08-06 13:03:06 +00:00
// we may learn about this now...
if ( need_many_cell_types ( ) & & o - > uWallOffset = = - 1 ) {
reset_raycaster ( ) ;
cast ( ) ;
return ;
}
2021-02-18 14:47:45 +00:00
GLERR ( " uniform mediump start " ) ;
2021-05-17 12:21:29 +00:00
if ( ! o ) { cast ( ) ; return ; }
2021-02-18 14:47:45 +00:00
uniform2 ( o - > uStartid , rmap - > enc ( rmap - > ids [ cs ] , 0 ) ) ;
2020-03-29 15:35:42 +00:00
}
2019-10-26 13:33:57 +00:00
2020-10-15 14:37:43 +00:00
# if CAP_VERTEXBUFFER
glhr : : bindbuffer_vertex ( screen ) ;
glVertexAttribPointer ( hr : : aPosition , 4 , GL_FLOAT , GL_FALSE , sizeof ( glvertex ) , 0 ) ;
# else
2019-10-21 22:37:57 +00:00
glVertexAttribPointer ( hr : : aPosition , 4 , GL_FLOAT , GL_FALSE , sizeof ( glvertex ) , & screen [ 0 ] ) ;
2020-10-15 14:37:43 +00:00
# endif
2019-10-26 12:11:44 +00:00
if ( ray : : comparison_mode )
glhr : : set_depthtest ( false ) ;
else {
glhr : : set_depthtest ( true ) ;
glhr : : set_depthwrite ( true ) ;
}
2019-10-21 22:37:57 +00:00
glBlendFunc ( GL_SRC_ALPHA , GL_ONE_MINUS_SRC_ALPHA ) ;
2022-08-05 17:22:01 +00:00
if ( ! floor_textures ) {
println ( hlog , " make_floor_textures called " ) ;
make_floor_textures ( ) ;
check_cgi ( ) ;
}
2019-10-26 08:32:44 +00:00
glActiveTexture ( GL_TEXTURE0 + 0 ) ;
glBindTexture ( GL_TEXTURE_2D , floor_textures - > renderedTexture ) ;
2020-10-15 14:37:43 +00:00
GLERR ( " bind " ) ;
2019-10-21 22:37:57 +00:00
glDrawArrays ( GL_TRIANGLES , 0 , 6 ) ;
GLERR ( " finish " ) ;
}
2020-07-07 20:16:38 +00:00
EX namespace volumetric {
EX bool on ;
EX map < cell * , color_t > vmap ;
int intensity = 16 ;
EX void enable ( ) {
if ( ! on ) {
on = true ;
reset_raycaster ( ) ;
}
}
EX void random_fog ( ) {
enable ( ) ;
for ( cell * c : currentmap - > allcells ( ) )
vmap [ c ] = ( ( rand ( ) % 0x1000000 ) < < 8 ) | intensity ;
}
EX void menu ( ) {
cmode = sm : : SIDE | sm : : MAYDARK ;
2022-07-05 09:51:06 +00:00
gamescreen ( ) ;
2020-07-07 20:16:38 +00:00
dialog : : init ( XLAT ( " volumetric raycasting " ) ) ;
2020-07-12 18:54:29 +00:00
if ( ! cheater ) {
dialog : : addItem ( XLAT ( " enable the cheat mode for additional options " ) , ' X ' ) ;
dialog : : add_action ( enable_cheat ) ;
dialog : : addBack ( ) ;
dialog : : display ( ) ;
return ;
}
2020-07-07 20:16:38 +00:00
dialog : : addBoolItem ( XLAT ( " active " ) , on , ' a ' ) ;
dialog : : add_action ( [ & ] {
on = ! on ;
reset_raycaster ( ) ;
} ) ;
dialog : : addSelItem ( XLAT ( " intensity of random coloring " ) , its ( intensity ) , ' i ' ) ;
dialog : : add_action ( [ ] {
dialog : : editNumber ( intensity , 0 , 255 , 5 , 15 , " " , " " ) ;
dialog : : reaction = random_fog ;
} ) ;
dialog : : addItem ( XLAT ( " color randomly " ) , ' r ' ) ;
dialog : : add_action ( random_fog ) ;
dialog : : addColorItem ( " color cell under cursor " , vmap . count ( centerover ) ? vmap [ centerover ] : 0 , ' c ' ) ;
dialog : : add_action ( [ & ] {
enable ( ) ;
dialog : : openColorDialog ( vmap [ centerover ] ) ;
dialog : : dialogflags | = sm : : SIDE ;
} ) ;
dialog : : addColorItem ( " color cell under player " , vmap . count ( cwt . at ) ? vmap [ cwt . at ] : 0 , ' p ' ) ;
dialog : : add_action ( [ & ] {
enable ( ) ;
dialog : : openColorDialog ( vmap [ cwt . at ] ) ;
dialog : : dialogflags | = sm : : SIDE ;
} ) ;
dialog : : addBreak ( 150 ) ;
dialog : : addHelp ( " This fills all the cells with glowing fog, for cool visualizations " ) ;
dialog : : addBreak ( 150 ) ;
dialog : : addBack ( ) ;
dialog : : display ( ) ;
}
EX }
2019-10-25 12:50:48 +00:00
EX void configure ( ) {
cmode = sm : : SIDE | sm : : MAYDARK ;
2022-07-05 09:51:06 +00:00
gamescreen ( ) ;
2019-10-25 12:50:48 +00:00
dialog : : init ( XLAT ( " raycasting configuration " ) ) ;
dialog : : addBoolItem ( XLAT ( " available in current geometry " ) , available ( ) , 0 ) ;
dialog : : addBoolItem ( XLAT ( " use raycasting? " ) , want_use = = 2 ? true : in_use , ' u ' ) ;
if ( want_use = = 1 ) dialog : : lastItem ( ) . value = XLAT ( " SMART " ) ;
dialog : : add_action ( [ ] {
want_use + + ; want_use % = 3 ;
} ) ;
dialog : : addBoolItem_action ( XLAT ( " comparison mode " ) , comparison_mode , ' c ' ) ;
2019-10-25 13:05:17 +00:00
dialog : : addSelItem ( XLAT ( " exponential range " ) , fts ( exp_decay_current ( ) ) , ' r ' ) ;
dialog : : add_action ( [ & ] {
2019-11-08 14:00:09 +00:00
dialog : : editNumber ( exp_decay_current ( ) , 0 , 40 , 0.25 , 5 , XLAT ( " exponential range " ) ,
2019-10-25 13:05:17 +00:00
XLAT ( " brightness formula: max(1-d/sightrange, s*exp(-d/r)) " )
) ;
} ) ;
dialog : : addSelItem ( XLAT ( " exponential start " ) , fts ( exp_start ) , ' s ' ) ;
dialog : : add_action ( [ & ] {
dialog : : editNumber ( exp_start , 0 , 1 , 0.1 , 1 , XLAT ( " exponential start " ) ,
XLAT ( " brightness formula: max(1-d/sightrange, s*exp(-d/r)) \n " )
) ;
} ) ;
2019-11-08 13:59:19 +00:00
if ( hard_limit < NO_LIMIT )
dialog : : addSelItem ( XLAT ( " hard limit " ) , fts ( hard_limit ) , ' H ' ) ;
else
dialog : : addBoolItem ( XLAT ( " hard limit " ) , false , ' H ' ) ;
dialog : : add_action ( [ & ] {
if ( hard_limit > = NO_LIMIT ) hard_limit = 10 ;
dialog : : editNumber ( hard_limit , 0 , 100 , 1 , 10 , XLAT ( " hard limit " ) , " " ) ;
dialog : : reaction = reset_raycaster ;
dialog : : extra_options = [ ] {
dialog : : addItem ( " no limit " , ' N ' ) ;
dialog : : add_action ( [ ] { hard_limit = NO_LIMIT ; reset_raycaster ( ) ; } ) ;
} ;
} ) ;
2019-10-27 00:23:20 +00:00
if ( ! nil ) {
dialog : : addSelItem ( XLAT ( " reflective walls " ) , fts ( reflect_val ) , ' R ' ) ;
dialog : : add_action ( [ & ] {
dialog : : editNumber ( reflect_val , 0 , 1 , 0.1 , 0 , XLAT ( " reflective walls " ) , " " ) ;
dialog : : reaction = reset_raycaster ;
} ) ;
}
2019-10-25 13:18:31 +00:00
2021-02-07 19:39:39 +00:00
if ( is_stepbased ( ) ) {
2019-10-25 22:43:15 +00:00
dialog : : addSelItem ( XLAT ( " max step " ) , fts ( maxstep_current ( ) ) , ' x ' ) ;
dialog : : add_action ( [ ] {
2021-02-07 19:39:39 +00:00
auto & ms = maxstep_current ( ) ;
dialog : : editNumber ( maxstep_current ( ) , 1e-6 , 1 , .1 ,
& ms = = & maxstep_pro ? .05 :
& ms = = & maxstep_nil ? .1 : .5 ,
XLAT ( " max step " ) , " affects the precision of solving the geodesic equation in Solv " ) ;
2019-10-25 22:43:15 +00:00
dialog : : scaleLog ( ) ;
dialog : : bound_low ( 1e-9 ) ;
dialog : : reaction = reset_raycaster ;
} ) ;
dialog : : addSelItem ( XLAT ( " min step " ) , fts ( minstep ) , ' n ' ) ;
dialog : : add_action ( [ ] {
2019-10-26 13:42:38 +00:00
dialog : : editNumber ( minstep , 1e-6 , 1 , 0.1 , 0.001 , XLAT ( " min step " ) , " how precisely should we find out when do cross the cell boundary " ) ;
2019-10-25 22:43:15 +00:00
dialog : : scaleLog ( ) ;
dialog : : bound_low ( 1e-9 ) ;
dialog : : reaction = reset_raycaster ;
} ) ;
}
2020-07-07 20:16:38 +00:00
2021-05-24 09:42:46 +00:00
dialog : : addBoolItem ( XLAT ( " volumetric raycasting " ) , volumetric : : on , ' v ' ) ;
2020-07-07 20:16:38 +00:00
dialog : : add_action_push ( volumetric : : menu ) ;
2019-10-25 13:18:31 +00:00
dialog : : addSelItem ( XLAT ( " iterations " ) , its ( max_iter_current ( ) ) , ' s ' ) ;
dialog : : add_action ( [ & ] {
2019-10-26 13:42:38 +00:00
dialog : : editNumber ( max_iter_current ( ) , 0 , 600 , 1 , 60 , XLAT ( " iterations " ) , " in H3/H2xE/E3 this is the number of cell boundaries; in nonisotropic, the number of simulation steps " ) ;
2019-10-25 22:43:15 +00:00
dialog : : reaction = reset_raycaster ;
2019-10-25 13:18:31 +00:00
} ) ;
2019-10-25 22:12:30 +00:00
dialog : : addSelItem ( XLAT ( " max cells " ) , its ( max_cells ) , ' s ' ) ;
dialog : : add_action ( [ & ] {
dialog : : editNumber ( max_cells , 16 , 131072 , 0.1 , 4096 , XLAT ( " max cells " ) , " " ) ;
dialog : : scaleLog ( ) ;
dialog : : extra_options = [ ] {
dialog : : addBoolItem_action ( " generate " , rays_generate , ' G ' ) ;
2019-10-25 22:17:58 +00:00
dialog : : addColorItem ( XLAT ( " out-of-range color " ) , color_out_of_range , ' X ' ) ;
dialog : : add_action ( [ ] {
dialog : : openColorDialog ( color_out_of_range ) ;
dialog : : dialogflags | = sm : : SIDE ;
} ) ;
2019-10-25 22:12:30 +00:00
} ;
} ) ;
2022-10-26 22:02:48 +00:00
dialog : : addBoolItem_action ( XLAT ( " the map is fixed (improves performance) " ) , ray : : fixed_map , ' F ' ) ;
2020-05-16 08:55:38 +00:00
2020-11-14 12:53:19 +00:00
if ( gms_array_size > gms_limit & & ray : : in_use ) {
2020-05-16 08:55:38 +00:00
dialog : : addBreak ( 100 ) ;
2020-11-14 12:53:19 +00:00
dialog : : addHelp ( XLAT ( " unfortunately this honeycomb is too complex for the current implementation (%1>%2) " , its ( gms_array_size ) , its ( gms_limit ) ) ) ;
2020-05-16 08:55:38 +00:00
}
2019-11-03 12:36:06 +00:00
edit_levellines ( ' L ' ) ;
2019-10-25 13:05:17 +00:00
2019-10-25 12:50:48 +00:00
dialog : : addBack ( ) ;
dialog : : display ( ) ;
}
2019-10-25 22:12:40 +00:00
# if CAP_COMMANDLINE
int readArgs ( ) {
using namespace arg ;
if ( 0 ) ;
else if ( argis ( " -ray-do " ) ) {
PHASEFROM ( 2 ) ;
want_use = 2 ;
}
else if ( argis ( " -ray-dont " ) ) {
PHASEFROM ( 2 ) ;
want_use = 0 ;
}
else if ( argis ( " -ray-smart " ) ) {
PHASEFROM ( 2 ) ;
want_use = 1 ;
}
2020-05-08 19:20:33 +00:00
else if ( argis ( " -ray-range " ) ) {
PHASEFROM ( 2 ) ;
2020-07-03 12:55:16 +00:00
shift_arg_formula ( exp_start , reset_raycaster ) ;
shift_arg_formula ( exp_decay_current ( ) , reset_raycaster ) ;
2020-05-27 23:50:53 +00:00
}
else if ( argis ( " -ray-hard " ) ) {
PHASEFROM ( 2 ) ;
shift_arg_formula ( hard_limit ) ;
2020-05-08 19:20:33 +00:00
}
2019-11-02 21:31:50 +00:00
else if ( argis ( " -ray-out " ) ) {
2021-10-17 07:18:31 +00:00
PHASEFROM ( 2 ) ; shift ( ) ; color_out_of_range = argcolor ( 32 ) ;
2019-11-02 21:31:50 +00:00
}
2019-10-28 16:33:54 +00:00
else if ( argis ( " -ray-comp " ) ) {
PHASEFROM ( 2 ) ;
comparison_mode = true ;
}
2020-05-08 19:20:33 +00:00
else if ( argis ( " -ray-sol " ) ) {
PHASEFROM ( 2 ) ;
shift ( ) ; max_iter_sol = argi ( ) ;
shift_arg_formula ( maxstep_sol , reset_raycaster ) ;
reset_raycaster ( ) ;
}
2020-07-03 12:55:16 +00:00
else if ( argis ( " -ray-iter " ) ) {
PHASEFROM ( 2 ) ;
shift ( ) ; max_iter_current ( ) = argi ( ) ;
}
2020-08-22 22:11:47 +00:00
else if ( argis ( " -ray-step " ) ) {
PHASEFROM ( 2 ) ;
println ( hlog , " maxstep_current() is " , maxstep_current ( ) ) ;
shift_arg_formula ( maxstep_current ( ) ) ;
}
2019-10-25 22:12:40 +00:00
else if ( argis ( " -ray-cells " ) ) {
PHASEFROM ( 2 ) ; shift ( ) ;
rays_generate = true ;
max_cells = argi ( ) ;
}
2019-10-27 00:23:20 +00:00
else if ( argis ( " -ray-reflect " ) ) {
PHASEFROM ( 2 ) ;
2020-07-03 12:55:16 +00:00
shift_arg_formula ( reflect_val , reset_raycaster ) ;
2019-10-27 00:23:20 +00:00
}
2019-10-25 22:12:40 +00:00
else if ( argis ( " -ray-cells-no " ) ) {
PHASEFROM ( 2 ) ; shift ( ) ;
rays_generate = false ;
max_cells = argi ( ) ;
}
2020-07-08 10:43:30 +00:00
else if ( argis ( " -ray-random " ) ) {
start_game ( ) ;
shift ( ) ; volumetric : : intensity = argi ( ) ;
volumetric : : random_fog ( ) ;
}
else if ( argis ( " -ray-cursor " ) ) {
start_game ( ) ;
volumetric : : enable ( ) ;
2021-10-17 07:18:31 +00:00
shift ( ) ; volumetric : : vmap [ centerover ] = argcolor ( 32 ) ;
2020-07-08 10:43:30 +00:00
}
2019-10-25 22:12:40 +00:00
else return 1 ;
return 0 ;
}
2021-02-23 04:33:35 +00:00
auto hook = addHook ( hooks_args , 100 , readArgs )
+ addHook ( hooks_clearmemory , 40 , [ ] { rmap = { } ; } ) ;
2019-11-23 22:51:52 +00:00
# endif
# if CAP_CONFIG
2019-11-16 00:40:14 +00:00
void addconfig ( ) {
2021-02-01 00:45:10 +00:00
param_f ( exp_start , " ray_exp_start " ) ;
param_f ( exp_decay_exp , " ray_exp_decay_exp " ) ;
param_f ( maxstep_sol , " ray_maxstep_sol " ) ;
param_f ( maxstep_nil , " ray_maxstep_nil " ) ;
param_f ( minstep , " ray_minstep " ) ;
param_f ( reflect_val , " ray_reflect_val " ) ;
param_f ( hard_limit , " ray_hard_limit " ) ;
2019-11-16 00:40:14 +00:00
addsaver ( want_use , " ray_want_use " ) ;
2021-02-18 14:48:00 +00:00
param_f ( exp_decay_poly , " ray_exp_decay_poly " ) ;
2019-11-16 00:40:14 +00:00
addsaver ( max_iter_iso , " ray_max_iter_iso " ) ;
addsaver ( max_iter_sol , " ray_max_iter_sol " ) ;
2021-10-09 15:05:31 +00:00
param_i ( max_cells , " ray_max_cells " ) ;
2019-11-16 00:40:14 +00:00
addsaver ( rays_generate , " ray_generate " ) ;
2021-02-18 14:47:45 +00:00
param_b ( fixed_map , " ray_fixed_map " ) ;
2022-09-14 21:38:45 +00:00
param_i ( max_wall_offset , " max_wall_offset " ) ;
param_i ( max_celltype , " max_celltype " ) ;
2019-11-16 00:40:14 +00:00
}
2019-11-23 22:51:52 +00:00
auto hookc = addHook ( hooks_configfile , 100 , addconfig ) ;
2019-10-25 22:12:40 +00:00
# endif
2019-11-23 22:51:52 +00:00
2019-11-03 13:19:11 +00:00
# endif
2019-10-25 22:12:40 +00:00
2020-03-21 09:49:07 +00:00
# if !CAP_RAY
2019-11-03 13:19:11 +00:00
EX always_false in_use ;
EX always_false comparison_mode ;
EX void reset_raycaster ( ) { }
EX void cast ( ) { }
# endif
2019-10-25 12:50:48 +00:00
EX }
2019-10-21 22:37:57 +00:00
}