|
|
@@ -0,0 +1,360 @@
|
|
|
+// Filename: qpgeomVertexArrayData.cxx
|
|
|
+// Created by: drose (17Mar05)
|
|
|
+//
|
|
|
+////////////////////////////////////////////////////////////////////
|
|
|
+//
|
|
|
+// PANDA 3D SOFTWARE
|
|
|
+// Copyright (c) 2001 - 2004, Disney Enterprises, Inc. All rights reserved
|
|
|
+//
|
|
|
+// All use of this software is subject to the terms of the Panda 3d
|
|
|
+// Software license. You should have received a copy of this license
|
|
|
+// along with this source code; you will also find a current copy of
|
|
|
+// the license at http://etc.cmu.edu/panda3d/docs/license/ .
|
|
|
+//
|
|
|
+// To contact the maintainers of this program write to
|
|
|
+// [email protected] .
|
|
|
+//
|
|
|
+////////////////////////////////////////////////////////////////////
|
|
|
+
|
|
|
+#include "qpgeomVertexArrayData.h"
|
|
|
+#include "preparedGraphicsObjects.h"
|
|
|
+#include "bamReader.h"
|
|
|
+#include "bamWriter.h"
|
|
|
+#include "pset.h"
|
|
|
+
|
|
|
+TypeHandle qpGeomVertexArrayData::_type_handle;
|
|
|
+
|
|
|
+////////////////////////////////////////////////////////////////////
|
|
|
+// Function: qpGeomVertexArrayData::Default Constructor
|
|
|
+// Access: Private
|
|
|
+// Description: Constructs an invalid object. This is only used when
|
|
|
+// reading from the bam file.
|
|
|
+////////////////////////////////////////////////////////////////////
|
|
|
+qpGeomVertexArrayData::
|
|
|
+qpGeomVertexArrayData() {
|
|
|
+}
|
|
|
+
|
|
|
+////////////////////////////////////////////////////////////////////
|
|
|
+// Function: qpGeomVertexArrayData::Constructor
|
|
|
+// Access: Published
|
|
|
+// Description:
|
|
|
+////////////////////////////////////////////////////////////////////
|
|
|
+qpGeomVertexArrayData::
|
|
|
+qpGeomVertexArrayData(const qpGeomVertexArrayFormat *array_format,
|
|
|
+ qpGeomVertexArrayData::UsageHint usage_hint) :
|
|
|
+ _array_format(array_format),
|
|
|
+ _usage_hint(usage_hint)
|
|
|
+{
|
|
|
+}
|
|
|
+
|
|
|
+////////////////////////////////////////////////////////////////////
|
|
|
+// Function: qpGeomVertexArrayData::Copy Constructor
|
|
|
+// Access: Published
|
|
|
+// Description:
|
|
|
+////////////////////////////////////////////////////////////////////
|
|
|
+qpGeomVertexArrayData::
|
|
|
+qpGeomVertexArrayData(const qpGeomVertexArrayData ©) :
|
|
|
+ _array_format(copy._array_format),
|
|
|
+ _usage_hint(copy._usage_hint),
|
|
|
+ _cycler(copy._cycler)
|
|
|
+{
|
|
|
+}
|
|
|
+
|
|
|
+////////////////////////////////////////////////////////////////////
|
|
|
+// Function: qpGeomVertexArrayData::Copy Assignment Operator
|
|
|
+// Access: Private
|
|
|
+// Description: Directly copying ArrayData objects by assignment is
|
|
|
+// disallowed.
|
|
|
+////////////////////////////////////////////////////////////////////
|
|
|
+void qpGeomVertexArrayData::
|
|
|
+operator = (const qpGeomVertexArrayData &) {
|
|
|
+ nassertv(false);
|
|
|
+}
|
|
|
+
|
|
|
+////////////////////////////////////////////////////////////////////
|
|
|
+// Function: qpGeomVertexArrayData::Destructor
|
|
|
+// Access: Published, Virtual
|
|
|
+// Description:
|
|
|
+////////////////////////////////////////////////////////////////////
|
|
|
+qpGeomVertexArrayData::
|
|
|
+~qpGeomVertexArrayData() {
|
|
|
+ release_all();
|
|
|
+}
|
|
|
+
|
|
|
+
|
|
|
+////////////////////////////////////////////////////////////////////
|
|
|
+// Function: qpGeomVertexArrayData::set_num_vertices
|
|
|
+// Access: Published
|
|
|
+// Description: Sets the length of the array to n vertices.
|
|
|
+// Normally, you would not call this directly, since all
|
|
|
+// of the arrays in a particular GeomVertexData must
|
|
|
+// have the same number of vertices; instead, call
|
|
|
+// GeomVertexData::set_num_vertices().
|
|
|
+//
|
|
|
+// The return value is true if the number of vertices
|
|
|
+// was changed, false if the object already contained n
|
|
|
+// vertices (or if there was some error).
|
|
|
+////////////////////////////////////////////////////////////////////
|
|
|
+bool qpGeomVertexArrayData::
|
|
|
+set_num_vertices(int n) {
|
|
|
+ CDWriter cdata(_cycler);
|
|
|
+
|
|
|
+ int stride = _array_format->get_stride();
|
|
|
+ int delta = n - (cdata->_data.size() / stride);
|
|
|
+
|
|
|
+ if (delta != 0) {
|
|
|
+ if (cdata->_data.get_ref_count() > 1) {
|
|
|
+ // Copy-on-write: the data is already reffed somewhere else,
|
|
|
+ // so we're just going to make a copy.
|
|
|
+ PTA_uchar new_data;
|
|
|
+ new_data.reserve(n * stride);
|
|
|
+ new_data.insert(new_data.end(), n * stride, uchar());
|
|
|
+ memcpy(new_data, cdata->_data,
|
|
|
+ min((size_t)(n * stride), cdata->_data.size()));
|
|
|
+ cdata->_data = new_data;
|
|
|
+
|
|
|
+ } else {
|
|
|
+ // We've got the only reference to the data, so we can change
|
|
|
+ // it directly.
|
|
|
+ if (delta > 0) {
|
|
|
+ cdata->_data.insert(cdata->_data.end(), delta * stride, uchar());
|
|
|
+
|
|
|
+ } else {
|
|
|
+ cdata->_data.erase(cdata->_data.begin() + n * stride,
|
|
|
+ cdata->_data.end());
|
|
|
+ }
|
|
|
+ }
|
|
|
+
|
|
|
+ ++(cdata->_modified);
|
|
|
+
|
|
|
+ return true;
|
|
|
+ }
|
|
|
+
|
|
|
+ return false;
|
|
|
+}
|
|
|
+
|
|
|
+////////////////////////////////////////////////////////////////////
|
|
|
+// Function: qpGeomVertexArrayData::prepare
|
|
|
+// Access: Published
|
|
|
+// Description: Indicates that the data should be enqueued to be
|
|
|
+// prepared in the indicated prepared_objects at the
|
|
|
+// beginning of the next frame. This will ensure the
|
|
|
+// data is already loaded into the GSG if it is expected
|
|
|
+// to be rendered soon.
|
|
|
+//
|
|
|
+// Use this function instead of prepare_now() to preload
|
|
|
+// datas from a user interface standpoint.
|
|
|
+////////////////////////////////////////////////////////////////////
|
|
|
+void qpGeomVertexArrayData::
|
|
|
+prepare(PreparedGraphicsObjects *prepared_objects) {
|
|
|
+ prepared_objects->enqueue_data(this);
|
|
|
+}
|
|
|
+
|
|
|
+////////////////////////////////////////////////////////////////////
|
|
|
+// Function: qpGeomVertexArrayData::prepare_now
|
|
|
+// Access: Public
|
|
|
+// Description: Creates a context for the data on the particular
|
|
|
+// GSG, if it does not already exist. Returns the new
|
|
|
+// (or old) DataContext. This assumes that the
|
|
|
+// GraphicsStateGuardian is the currently active
|
|
|
+// rendering context and that it is ready to accept new
|
|
|
+// datas. If this is not necessarily the case, you
|
|
|
+// should use prepare() instead.
|
|
|
+//
|
|
|
+// Normally, this is not called directly except by the
|
|
|
+// GraphicsStateGuardian; a data does not need to be
|
|
|
+// explicitly prepared by the user before it may be
|
|
|
+// rendered.
|
|
|
+////////////////////////////////////////////////////////////////////
|
|
|
+DataContext *qpGeomVertexArrayData::
|
|
|
+prepare_now(PreparedGraphicsObjects *prepared_objects,
|
|
|
+ GraphicsStateGuardianBase *gsg) {
|
|
|
+ Contexts::const_iterator ci;
|
|
|
+ ci = _contexts.find(prepared_objects);
|
|
|
+ if (ci != _contexts.end()) {
|
|
|
+ return (*ci).second;
|
|
|
+ }
|
|
|
+
|
|
|
+ DataContext *dc = prepared_objects->prepare_data_now(this, gsg);
|
|
|
+ if (dc != (DataContext *)NULL) {
|
|
|
+ _contexts[prepared_objects] = dc;
|
|
|
+ }
|
|
|
+ return dc;
|
|
|
+}
|
|
|
+
|
|
|
+////////////////////////////////////////////////////////////////////
|
|
|
+// Function: qpGeomVertexArrayData::release
|
|
|
+// Access: Public
|
|
|
+// Description: Frees the data context only on the indicated object,
|
|
|
+// if it exists there. Returns true if it was released,
|
|
|
+// false if it had not been prepared.
|
|
|
+////////////////////////////////////////////////////////////////////
|
|
|
+bool qpGeomVertexArrayData::
|
|
|
+release(PreparedGraphicsObjects *prepared_objects) {
|
|
|
+ Contexts::iterator ci;
|
|
|
+ ci = _contexts.find(prepared_objects);
|
|
|
+ if (ci != _contexts.end()) {
|
|
|
+ DataContext *dc = (*ci).second;
|
|
|
+ prepared_objects->release_data(dc);
|
|
|
+ return true;
|
|
|
+ }
|
|
|
+
|
|
|
+ // Maybe it wasn't prepared yet, but it's about to be.
|
|
|
+ return prepared_objects->dequeue_data(this);
|
|
|
+}
|
|
|
+
|
|
|
+////////////////////////////////////////////////////////////////////
|
|
|
+// Function: qpGeomVertexArrayData::release_all
|
|
|
+// Access: Public
|
|
|
+// Description: Frees the context allocated on all objects for which
|
|
|
+// the data has been declared. Returns the number of
|
|
|
+// contexts which have been freed.
|
|
|
+////////////////////////////////////////////////////////////////////
|
|
|
+int qpGeomVertexArrayData::
|
|
|
+release_all() {
|
|
|
+ // We have to traverse a copy of the _contexts list, because the
|
|
|
+ // PreparedGraphicsObjects object will call clear_prepared() in response
|
|
|
+ // to each release_data(), and we don't want to be modifying the
|
|
|
+ // _contexts list while we're traversing it.
|
|
|
+ Contexts temp = _contexts;
|
|
|
+ int num_freed = (int)_contexts.size();
|
|
|
+
|
|
|
+ Contexts::const_iterator ci;
|
|
|
+ for (ci = temp.begin(); ci != temp.end(); ++ci) {
|
|
|
+ PreparedGraphicsObjects *prepared_objects = (*ci).first;
|
|
|
+ DataContext *dc = (*ci).second;
|
|
|
+ prepared_objects->release_data(dc);
|
|
|
+ }
|
|
|
+
|
|
|
+ // Now that we've called release_data() on every known context,
|
|
|
+ // the _contexts list should have completely emptied itself.
|
|
|
+ nassertr(_contexts.empty(), num_freed);
|
|
|
+
|
|
|
+ return num_freed;
|
|
|
+}
|
|
|
+
|
|
|
+////////////////////////////////////////////////////////////////////
|
|
|
+// Function: qpGeomVertexArrayData::clear_prepared
|
|
|
+// Access: Private
|
|
|
+// Description: Removes the indicated PreparedGraphicsObjects table
|
|
|
+// from the data array's table, without actually
|
|
|
+// releasing the data array. This is intended to be
|
|
|
+// called only from
|
|
|
+// PreparedGraphicsObjects::release_data(); it should
|
|
|
+// never be called by user code.
|
|
|
+////////////////////////////////////////////////////////////////////
|
|
|
+void qpGeomVertexArrayData::
|
|
|
+clear_prepared(PreparedGraphicsObjects *prepared_objects) {
|
|
|
+ Contexts::iterator ci;
|
|
|
+ ci = _contexts.find(prepared_objects);
|
|
|
+ if (ci != _contexts.end()) {
|
|
|
+ _contexts.erase(ci);
|
|
|
+ } else {
|
|
|
+ // If this assertion fails, clear_prepared() was given a
|
|
|
+ // prepared_objects which the data array didn't know about.
|
|
|
+ nassertv(false);
|
|
|
+ }
|
|
|
+}
|
|
|
+
|
|
|
+////////////////////////////////////////////////////////////////////
|
|
|
+// Function: qpGeomVertexArrayData::register_with_read_factory
|
|
|
+// Access: Public, Static
|
|
|
+// Description: Tells the BamReader how to create objects of type
|
|
|
+// qpGeomVertexArrayData.
|
|
|
+////////////////////////////////////////////////////////////////////
|
|
|
+void qpGeomVertexArrayData::
|
|
|
+register_with_read_factory() {
|
|
|
+ BamReader::get_factory()->register_factory(get_class_type(), make_from_bam);
|
|
|
+}
|
|
|
+
|
|
|
+////////////////////////////////////////////////////////////////////
|
|
|
+// Function: qpGeomVertexArrayData::write_datagram
|
|
|
+// Access: Public, Virtual
|
|
|
+// Description: Writes the contents of this object to the datagram
|
|
|
+// for shipping out to a Bam file.
|
|
|
+////////////////////////////////////////////////////////////////////
|
|
|
+void qpGeomVertexArrayData::
|
|
|
+write_datagram(BamWriter *manager, Datagram &dg) {
|
|
|
+ TypedWritable::write_datagram(manager, dg);
|
|
|
+
|
|
|
+ manager->write_cdata(dg, _cycler);
|
|
|
+}
|
|
|
+
|
|
|
+////////////////////////////////////////////////////////////////////
|
|
|
+// Function: qpGeomVertexArrayData::make_from_bam
|
|
|
+// Access: Protected, Static
|
|
|
+// Description: This function is called by the BamReader's factory
|
|
|
+// when a new object of type qpGeomVertexArrayData is encountered
|
|
|
+// in the Bam file. It should create the qpGeomVertexArrayData
|
|
|
+// and extract its information from the file.
|
|
|
+////////////////////////////////////////////////////////////////////
|
|
|
+TypedWritable *qpGeomVertexArrayData::
|
|
|
+make_from_bam(const FactoryParams ¶ms) {
|
|
|
+ qpGeomVertexArrayData *object = new qpGeomVertexArrayData;
|
|
|
+ DatagramIterator scan;
|
|
|
+ BamReader *manager;
|
|
|
+
|
|
|
+ parse_params(params, scan, manager);
|
|
|
+ object->fillin(scan, manager);
|
|
|
+
|
|
|
+ return object;
|
|
|
+}
|
|
|
+
|
|
|
+////////////////////////////////////////////////////////////////////
|
|
|
+// Function: qpGeomVertexArrayData::fillin
|
|
|
+// Access: Protected
|
|
|
+// Description: This internal function is called by make_from_bam to
|
|
|
+// read in all of the relevant data from the BamFile for
|
|
|
+// the new qpGeomVertexArrayData.
|
|
|
+////////////////////////////////////////////////////////////////////
|
|
|
+void qpGeomVertexArrayData::
|
|
|
+fillin(DatagramIterator &scan, BamReader *manager) {
|
|
|
+ TypedWritable::fillin(scan, manager);
|
|
|
+
|
|
|
+ manager->read_cdata(scan, _cycler);
|
|
|
+}
|
|
|
+
|
|
|
+////////////////////////////////////////////////////////////////////
|
|
|
+// Function: qpGeomVertexArrayData::CData::make_copy
|
|
|
+// Access: Public, Virtual
|
|
|
+// Description:
|
|
|
+////////////////////////////////////////////////////////////////////
|
|
|
+CycleData *qpGeomVertexArrayData::CData::
|
|
|
+make_copy() const {
|
|
|
+ return new CData(*this);
|
|
|
+}
|
|
|
+
|
|
|
+////////////////////////////////////////////////////////////////////
|
|
|
+// Function: qpGeomVertexArrayData::CData::write_datagram
|
|
|
+// Access: Public, Virtual
|
|
|
+// Description: Writes the contents of this object to the datagram
|
|
|
+// for shipping out to a Bam file.
|
|
|
+////////////////////////////////////////////////////////////////////
|
|
|
+void qpGeomVertexArrayData::CData::
|
|
|
+write_datagram(BamWriter *manager, Datagram &dg) const {
|
|
|
+}
|
|
|
+
|
|
|
+////////////////////////////////////////////////////////////////////
|
|
|
+// Function: qpGeomVertexArrayData::CData::complete_pointers
|
|
|
+// Access: Public, Virtual
|
|
|
+// Description: Receives an array of pointers, one for each time
|
|
|
+// manager->read_pointer() was called in fillin().
|
|
|
+// Returns the number of pointers processed.
|
|
|
+////////////////////////////////////////////////////////////////////
|
|
|
+int qpGeomVertexArrayData::CData::
|
|
|
+complete_pointers(TypedWritable **p_list, BamReader *manager) {
|
|
|
+ int pi = CycleData::complete_pointers(p_list, manager);
|
|
|
+
|
|
|
+ return pi;
|
|
|
+}
|
|
|
+
|
|
|
+////////////////////////////////////////////////////////////////////
|
|
|
+// Function: qpGeomVertexArrayData::CData::fillin
|
|
|
+// Access: Public, Virtual
|
|
|
+// Description: This internal function is called by make_from_bam to
|
|
|
+// read in all of the relevant data from the BamFile for
|
|
|
+// the new qpGeomVertexArrayData.
|
|
|
+////////////////////////////////////////////////////////////////////
|
|
|
+void qpGeomVertexArrayData::CData::
|
|
|
+fillin(DatagramIterator &scan, BamReader *manager) {
|
|
|
+}
|