/usr/share/psychtoolbox-3/PsychOpenGL/moglDrawDots3D.m is in psychtoolbox-3-common 3.0.9+svn2579.dfsg1-1.
This file is owned by root:root, with mode 0o644.
The actual contents of the file can be viewed below.
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168 169 170 171 172 173 174 175 176 177 178 179 180 181 182 183 184 185 186 187 188 189 190 191 192 193 194 195 196 197 198 199 200 201 202 203 204 205 206 207 208 209 210 211 212 213 214 215 216 217 218 219 220 221 222 223 224 225 226 227 228 229 230 231 232 233 234 235 236 237 238 239 240 241 242 243 244 245 246 247 248 249 250 251 252 253 254 255 256 257 258 259 260 261 262 263 264 265 266 267 268 269 270 271 272 273 274 275 276 277 278 279 280 281 282 283 284 285 286 287 288 289 290 291 292 293 294 295 296 297 298 | function moglDrawDots3D(windowPtr, xyz, dotdiameter, dotcolor, center3D, dot_type, glslshader)
% Draw a large number of dots in 3D very efficiently.
%
% Usage: moglDrawDots3D(windowPtr, xyz [,dotdiameter] [,dotcolor] [,center3D] [,dot_type] [, glslshader]);
%
% This function is the 3D equivalent of the Screen('DrawDots') subfunction
% for fast drawing of 2D dots. It has mostly the same paramters as that
% function, but extended into the 3D domain. It accepts a subset of the
% parameters for that function, ie., it is less liberal in what it accepts,
% in order to allow for simpler code and maximum efficiency.
%
% As a bonus, it accepts one additional parameter 'glslshader', the
% optional handle to a GLSL shading program, e.g., as loaded from
% filesystem via LoadGLSLProgram().
%
% The routine will draw into the 3D OpenGL userspace rendering context of
% onscreen window or offscreen window (or texture) 'windowPtr'. It will
% automatically switch to that window if it isn't already active in 3D
% mode, and it will restore the drawing target to whatever was set before
% invocation in whatever mode (2D or 3D). This is a convenience feature for
% lazy users that mostly draw in 2D. If you intend to draw more stuff in 3D
% for a given frame, then you should switch your targetwindow 'windowPtr'
% into 3D mode manually via Screen('BeginOpenGL') yourself beforehand. This
% will avoid redundant and expensive context switches and increase the
% execution speed of your code!
%
% Parameters and their meaning:
%
% 'windowPtr' Handle of window or texture to draw into.
% 'xyz' A 3-by-n or 4-by-n matrix of n dots to draw. Each column defines
% one dot to draw, either as 3D position (x,y,z) or 4D position (x,y,z,w).
% Must be a double matrix!
%
% 'dotdiameter' optional: Either a single scalar spec of dot diameter, or a
% vector of as many dotdiameters as dots 'n', or left out. If left out, a
% dot diameter of 1.0 pixels will be used. Drawing of dots of different
% sizes is much less efficient than drawing of dots of identical sizes! Try
% to group many dots of identical size into separate calls to this function
% for best performance!
%
% 'dotcolor' optional: Either a 3 or 4-component [R,G,B] or [R,G,B,A] color
% touple with a common drawing color, or a 3-by-n or 4-by-n matrix of
% colors, one [R;G;B;A] column for each individual dot. A common color for
% all dots is faster.
%
% 'dot_type' optional: A setting of zero will draw rectangular dots, a
% setting of 1 will draw round dots, a setting of 2 will draw round dots of
% extra high quality if the hardware supports that. For anti-aliased dots
% you must select a setting of 1 or 2 and enable alpha blending as well.
%
% 'glslshader' optional: If omitted, shading state is not changed. If set
% to zero, then the standard fixed function OpenGL pipeline is used, like
% in Screen('DrawDots') (under most circumstances). If a positive
% glslshader handle to a GLSL shading program object is provided, that
% shader program will be used. You can use this, e.g., to bind a custom vertex
% shader to perform complex per-dot calculations very fast on the GPU.
%
% See
% History:
% 03/01/2009 mk Written.
% Need global GL definitions:
global GL;
% Child protection:
if isempty(GL)
error('Need OpenGL mode to be enabled, which is not the case! Call InitializeMatlabOpenGL at start of your script first!');
end
if nargin < 1 || isempty(windowPtr)
error('"windowPtr" window handle missing! This is required!');
end
if nargin < 2 || isempty(xyz)
% xyz dot position matrix is empty! Nothing to do for us:
return;
end
if ndims(xyz)~=2
error('"xyz" matrix with 3D dot positions is not a 2D matrix! This is required!');
end
% Want single xyz vector as a 3 or 4 row, 1 column vector:
if (size(xyz, 1) == 1) && (ismember(size(xyz, 2), [3,4]))
xyz = transpose(xyz);
end
nvc = size(xyz, 1);
nrdots = size(xyz, 2);
if ~ismember(nvc, [3,4]) || nrdots < 1
error('"xyz" argument must have 3 or 4 rows for x,y,z or x,y,z,w components and at least 1 column for at least one dot to draw!');
end
if nargin < 3 || isempty(dotdiameter)
dotdiameter = 1;
end
nsizes = length(dotdiameter);
if ~isvector(dotdiameter) || (nsizes~=1 && nsizes~=nrdots)
error('"dotdiameter" argument must be a vector with same number of elements as dots to draw, or a single scalar value!');
end
if nargin < 4
dotcolor = [];
end
if ~isempty(dotcolor)
% Want single dotcolor vector as a 4 row, 1 column vector:
if (size(dotcolor, 1) == 1) && (ismember(size(dotcolor, 2), [3,4]))
dotcolor = transpose(dotcolor);
end
ncolors = size(dotcolor, 2);
ncolcomps = size(dotcolor, 1);
if ~ismember(ncolcomps, [3,4]) || (ncolors~=1 && ncolors~=nrdots)
error('"dotcolor" must be a matrix with 3 or 4 rows and at least 1 column, or as many columns as dots to draw!');
end
else
ncolors = 0;
end
if nargin < 5
% Default to "no center set":
center3D = [];
end
if nargin < 6 || isempty(dot_type)
% Default to "no point smoothing set":
dot_type = 0;
end
if nargin < 7
% Default to no change of shader bindings:
glslshader = [];
end
% Is the OpenGL userspace context for this 'windowPtr' active, as required?
[previouswin, IsOpenGLRendering] = Screen('GetOpenGLDrawMode');
PreIsOpenGLRendering = IsOpenGLRendering;
% Our target window windowPtr already active?
if previouswin ~= windowPtr
% No. Wrong window. OpenGL rendering for this window active?
if IsOpenGLRendering
% Yes. We need to disable OpenGL mode for that other window and
% switch to our window:
Screen('EndOpenGL', windowPtr);
% Our window is attached, but it is in 2D mode, not 3D mode yet:
IsOpenGLRendering = 0;
end
end
% Either a different window than ours is bound in 2D mode, then OpenGL
% rendering is not yet active and we need to switch to our window and to
% OpenGL rendering.
%
% Or we just switched from a different window in 3D mode to our window in
% 2D mode. Then we need to switch our window into 3D mode.
%
% In both cases, IsOpenGLRendering == false will indicate this.
%
% A third option is that our wanted window is already active and 3D OpenGL
% mode is already active. In that case IsOpenGLRendering == true and we
% don't need to do anything to switch modes:
if ~IsOpenGLRendering
% Either some window, or our window bound in 2D mode. Need to switch to
% our window in 3D mode:
Screen('BeginOpenGL', windowPtr);
end
% Ok our target window and userspace OpenGL rendering context is bound, we
% can setup and execute the actual drawing:
% 'center3D' argument specified?
if ~isempty(center3D)
% Yes. Backup current modelview matrix, then apply translation
% transform:
glMatrixMode(GL.MODELVIEW);
glPushMatrix;
glTranslated(center3D(1), center3D(2), center3D(3));
end
% Point smoothing wanted?
if dot_type > 0
glEnable(GL.POINT_SMOOTH);
if dot_type > 1
% A dot type of 2 requests for highest quality point smoothing:
glHint(GL.POINT_SMOOTH_HINT, GL.NICEST);
else
glHint(GL.POINT_SMOOTH_HINT, GL.DONT_CARE);
end
end
% Pass a pointer to the start of the point-coordinate array:
glVertexPointer(nvc, GL.DOUBLE, 0, xyz);
% Enable fast rendering of arrays:
glEnableClientState(GL.VERTEX_ARRAY);
% Multiple colors, one per dot, provided?
if ncolors > 0
if ncolors > 1
% Yes. Setup a color array for fast drawing:
glColorPointer(ncolcomps, GL.DOUBLE, 0, dotcolor);
glEnableClientState(GL.COLOR_ARRAY);
else
% No. Just set one single common color:
if ncolcomps == 4
glColor4dv(dotcolor);
else
glColor3dv(dotcolor);
end
end
end
% Change of shader binding requested?
if ~isempty(glslshader)
% Backup old shader binding:
oldShader = glGetIntegerv(GL.CURRENT_PROGRAM);
% Set new one:
glUseProgram(glslshader);
end
% dotdiameter per dot defined?
if nsizes > 1
% Yes :-( -- Need to iterate over all dots and set each dots size
% individually in immediate mode. This is sloooow:
for i=1:nrdots
% Set dotsize:
glPointSize(dotdiameter(i));
% Submit vertex for drawing of this dot:
glDrawArrays(GL.POINTS, i-1, 1);
end
else
% No :-) -- We can use a single call for fast batch-drawing of all
% points with a common size:
glPointSize(dotdiameter);
glDrawArrays(GL.POINTS, 0, nrdots);
end
if ~isempty(glslshader)
% Reset old shader binding:
glUseProgram(oldShader);
end
if ncolors > 1
% Disable color array for fast drawing:
glColorPointer(ncolcomps, GL.DOUBLE, 0, 0);
glDisableClientState(GL.COLOR_ARRAY);
end
% Disable fast rendering of arrays:
glDisableClientState(GL.VERTEX_ARRAY);
glVertexPointer(nvc, GL.DOUBLE, 0, 0);
if dot_type > 0
glDisable(GL.POINT_SMOOTH);
end
if ~isempty(center3D)
% Restore old modelview matrix from backup:
glPopMatrix;
end
% Reset dot size to 1.0:
glPointSize(1);
% Our work is done. If a different window than our target window was
% active, we'll switch back to that window and its state:
if previouswin ~= windowPtr
% Different window was active before our invocation. Need to disable
% our 3D mode and switch back to that window (in 2D mode):
Screen('EndOpenGL', previouswin);
% Was that window in 3D mode, i.e., OpenGL rendering for that window was active?
if PreIsOpenGLRendering
% Yes. We need to switch that window back into 3D OpenGL mode:
Screen('BeginOpenGL', previouswin);
end
else
% Our window was active beforehand. Was it in 2D mode? In that case we
% need to switch our window back to 2D mode. Otherwise we'll just stay
% in 3D mode:
if ~PreIsOpenGLRendering
% Was in 2D mode. We need to switch back to 2D:
Screen('EndOpenGL', windowPtr);
end
end
% Switchback complete. The graphics system is the same state as it was
% before our invocation.
return;
|