File: cuda_device_mesh.cpp

package info (click to toggle)
k3d 0.8.0.2-6
  • links: PTS, VCS
  • area: main
  • in suites: squeeze
  • size: 40,692 kB
  • ctags: 39,695
  • sloc: cpp: 171,303; ansic: 24,129; xml: 6,995; python: 5,796; makefile: 726; sh: 22
file content (316 lines) | stat: -rw-r--r-- 10,764 bytes parent folder | download | duplicates (5)
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
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
// K-3D
// Copyright (c) 1995-2008, Timothy M. Shead
//
// Contact: tshead@k-3d.com
//
// This program is free software; you can redistribute it and/or
// modify it under the terms of the GNU General Public
// License as published by the Free Software Foundation; either
// version 2 of the License, or (at your option) any later version.
//
// This program is distributed in the hope that it will be useful,
// but WITHOUT ANY WARRANTY; without even the implied warranty of
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
// General Public License for more details.
//
// You should have received a copy of the GNU General Public
// License along with this program; if not, write to the Free Software
// Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA  02111-1307  USA

/** \file
    \author Evan Lezar (evanlezar@gmail.com)
    \date 27 July 2008
*/


#include "cuda_device_mesh.h"

/// Constructors
cuda_device_mesh::cuda_device_mesh ( ):
	m_p_host_mesh(0),
	m_cuda_device_polyhedra()
{
	pdev_points_and_selection = 0;
    m_number_of_points = 0;
}

cuda_device_mesh::cuda_device_mesh ( const k3d::mesh& mesh ):
	m_p_host_mesh(&mesh),
	m_cuda_device_polyhedra()
{
	if ( mesh.polyhedra )
	{
		m_cuda_device_polyhedra.set_polyhedra(*mesh.polyhedra);
	}
    pdev_points_and_selection = 0;
    m_number_of_points = mesh.points->size();
}

/// Destructor - frees the allocated device pointers
cuda_device_mesh::~cuda_device_mesh()
{
    //k3d::log() << debug << "~cuda_device_mesh" << std::endl;
    free_device_memory((void*)pdev_points_and_selection);
}

/**
 * Set the mesh
 */
void cuda_device_mesh::set_mesh ( const k3d::mesh& input_mesh )
{
	m_p_host_mesh = &input_mesh;
	// TODO:  Set the points and polyhedra associated with the mesh
    set_points_and_selections( *input_mesh.points, *input_mesh.point_selection);
}

/**
 * Set the point and selection pointers
 */
void cuda_device_mesh::set_points_and_selections ( k3d::mesh::points_t input_points, k3d::mesh::selection_t input_selections )
{
	m_p_host_points = &input_points;
    m_number_of_points = input_points.size();
	m_p_host_selections = &input_selections;
}

/**
 * copy the specified mesh to the device
 */
void cuda_device_mesh::copy_to_device( k3d::uint32_t what_to_copy )
{
	float* host_points_and_selection = 0;
    if ( what_to_copy & (MESH_POINTS + MESH_SELECTION) )
    {
    	bool copy_points = what_to_copy & MESH_POINTS;
        bool copy_selection = what_to_copy & MESH_SELECTION;
    	// allocate the memory for the points and the point selection
    	allocate_device_memory((void**)&pdev_points_and_selection, (m_number_of_points)*4*sizeof(float));

    	// convert the point coordinates and selection to floats and copy to the device
    	// first allocate host memory
		host_points_and_selection = (float*) malloc (  m_number_of_points*4*sizeof(float) );

		for ( int point = 0 ; point < m_number_of_points ; ++point )
		{
            int float_index = point*4;
            if ( copy_points )
            {
                host_points_and_selection[float_index] = (float)((*m_p_host_mesh->points)[point][0]);
                host_points_and_selection[float_index+1] = (float)((*m_p_host_mesh->points)[point][1]);
                host_points_and_selection[float_index+2] = (float)((*m_p_host_mesh->points)[point][2]);
            }
            if ( copy_selection )
            {
                host_points_and_selection[float_index+3] = (float)((*m_p_host_mesh->point_selection)[point]);
            }
		}

		// copy the coordinates and selection to the device
		copy_from_host_to_device((void*)pdev_points_and_selection, (const void*)host_points_and_selection, m_number_of_points*4*sizeof(float));
    }

    // remember to also copy the device versions of polyhedra etc
    if ( what_to_copy & ALL_POLYHEDRA_INFO )
    {
    	m_cuda_device_polyhedra.copy_to_device( what_to_copy );
    }

    // free the temporary host data
	synchronize_threads();
	free ( host_points_and_selection );
}

/**
 * Copy the specified mesh from the device
 */
void cuda_device_mesh::copy_from_device( k3d::mesh& destination_mesh, k3d::uint32_t what_to_copy )
{
	k3d::mesh* p_output_mesh = &destination_mesh;
	float* host_points_and_selection = 0;
	if ( what_to_copy & (MESH_POINTS + MESH_SELECTION) )
    {
    	bool copy_points = what_to_copy & MESH_POINTS;
        bool copy_selection = what_to_copy & MESH_SELECTION;

    	// allocate temporary storage for the floats from the device
    	host_points_and_selection = (float*) malloc ( m_number_of_points*4*sizeof(float) );
    	copy_from_device_to_host( (void*) host_points_and_selection, (void*) pdev_points_and_selection, m_number_of_points*4*sizeof(float) );

        double* out_points;
        double* out_selection;

    	if ( copy_points )
        {
            // check to see if mesh's points exist
            if ( !(p_output_mesh->points) )
            {
                p_output_mesh->points.create(new k3d::mesh::points_t(m_number_of_points));
            }
            // check to see if the number of points has changed
            else if ( m_number_of_points  != p_output_mesh->points->size() )
            {
                p_output_mesh->points.create( new k3d::mesh::points_t ( m_number_of_points ) );
                p_output_mesh->point_selection.create ( new k3d::mesh::selection_t ( m_number_of_points ) );
            }
            out_points = (double*)&(p_output_mesh->points->front());
        }

        if ( copy_selection )
        {
    		if ( !(p_output_mesh->point_selection) )
    		{
    			p_output_mesh->point_selection.create(new k3d::mesh::selection_t(m_number_of_points));
    		}
            else if ( m_number_of_points  != p_output_mesh->point_selection->size() )
            {
                p_output_mesh->point_selection.create ( new k3d::mesh::selection_t ( m_number_of_points ) );
            }
            out_selection = (double*)&(p_output_mesh->point_selection->front());
        }

		synchronize_threads();
		for ( int point = 0 ; point < m_number_of_points ; ++point )
		{
            int float_index = point*4;
			int double_index = point*3;
            if ( copy_points )
            {
                out_points[double_index] = (double)(host_points_and_selection[float_index]);
                out_points[double_index+1] = (double)(host_points_and_selection[float_index+1]);
                out_points[double_index+2] = (double)(host_points_and_selection[float_index+2]);
            }
            if ( copy_selection )
            {
                out_selection[point] = (double)(host_points_and_selection[float_index+3]);
            }
		}
    }
    // copy the polyhedra data from the device

	if ( what_to_copy & ALL_POLYHEDRA_INFO )
	{
        if ( !p_output_mesh->polyhedra )
        {
            p_output_mesh->polyhedra.create();
        }

        m_cuda_device_polyhedra.copy_from_device(*(p_output_mesh->polyhedra), what_to_copy);
	}

    synchronize_threads();
    free ( host_points_and_selection );
}

/**
 * Allocate the memory for the mesh on the device - good for sources and output meshes
 */
void cuda_device_mesh::allocate_on_device( k3d::uint32_t what_to_allocate )
{
	if ( what_to_allocate & (MESH_POINTS + MESH_SELECTION) )
	{
		bool allocate_points = what_to_allocate & MESH_POINTS;
		bool allocate_selection = what_to_allocate & MESH_SELECTION;
		// allocate the memory for the points and the point selection
		allocate_device_memory((void**)&pdev_points_and_selection, (m_number_of_points)*4*sizeof(float));
	}

	// remember to also copy the device versions of polyhedra etc
	if ( what_to_allocate & ALL_POLYHEDRA_INFO )
	{
		m_cuda_device_polyhedra.allocate_on_device( what_to_allocate );
	}

	// free the temporary host data
	synchronize_threads();
}

/**
 * Ouput debug information for the data structure
 */
void cuda_device_mesh::output_debug_info()
{
    k3d::log() << debug << "=====================" << std::endl;
    k3d::log() << debug << "CUDA Device mesh info" << std::endl;
    k3d::log() << debug << "=====================" << std::endl;
    k3d::log() << debug << "pdev_points_and_selection: " << pdev_points_and_selection << std::endl;
    k3d::log() << debug << "m_number_of_points: " << m_number_of_points << std::endl;
    k3d::log() << debug << "m_p_host_mesh: " << m_p_host_mesh << std::endl;
    m_cuda_device_polyhedra.output_debug_info();
    k3d::log() << debug << "=====================" << std::endl;
}


cuda_device_polyhedra& cuda_device_mesh::get_device_polyhedra ()
{
    return m_cuda_device_polyhedra;
}

float* cuda_device_mesh::get_points_and_selection_pointer()
{
    return pdev_points_and_selection;
}

k3d::uint32_t* cuda_device_mesh::get_polyhedra_edge_point_indices_pointer()
{
    return m_cuda_device_polyhedra.get_per_edge_points_pointer();
}

k3d::uint32_t* cuda_device_mesh::get_polyhedra_clockwise_edge_point_indices_pointer()
{
    return m_cuda_device_polyhedra.get_per_edge_clockwise_edges_pointer();
}

k3d::uint32_t* cuda_device_mesh::get_polyhedra_loop_first_edges_pointer()
{
    return m_cuda_device_polyhedra.get_per_loop_first_edges_pointer();
}

void cuda_device_mesh::set_device_polyhedra( const k3d::mesh::polyhedra_t& host_polyhedra )
{

}

/**
 * Allocate space on the device for a new number of points and copy the data if required
 */
void cuda_device_mesh::resize_points_and_selection ( k3d::uint32_t new_number_of_points, float default_selection )
{
    float* new_pdev_point_and_selection;
    allocate_device_memory((void**)&new_pdev_point_and_selection, new_number_of_points*sizeof(float)*4);

    if ( pdev_points_and_selection != 0 )
    {
        copy_from_device_to_device((void*)new_pdev_point_and_selection, (const void*)pdev_points_and_selection, std::min(m_number_of_points, new_number_of_points)*4*sizeof(float));

        // initialize the selection to its default value
        if ( ( new_number_of_points > m_number_of_points ) & ( default_selection != 0 ) )
        {
            set_selection_value_entry ( new_pdev_point_and_selection + m_number_of_points*4, default_selection, new_number_of_points - m_number_of_points );
        }
        // free the memory allocated previously
        free_device_memory( pdev_points_and_selection );
    }

    synchronize_threads();
    pdev_points_and_selection = new_pdev_point_and_selection;
    m_number_of_points = new_number_of_points;
}

/**
 * Set all the selection values to 1
 */
void cuda_device_mesh::select_all_points()
{
	if ( pdev_points_and_selection )
	{
		set_selection_value_entry ( pdev_points_and_selection, 1.0, m_number_of_points );
	}
}

/**
 *  Get the number of points in the device mesh
 */
k3d::uint32_t cuda_device_mesh::get_number_of_points()
{
	return m_number_of_points;
}