blob: 06e7487acab564e304fe047bb8748630755f57d6 [file] [log] [blame]
daniel@transgaming.com4f39fd92010-03-08 20:26:45 +00001//
2// Copyright (c) 2002-2010 The ANGLE Project Authors. All rights reserved.
3// Use of this source code is governed by a BSD-style license that can be
4// found in the LICENSE file.
5//
6
7// Buffer.cpp: Implements the gl::Buffer class, representing storage of vertex and/or
8// index data. Implements GL buffer objects and related functionality.
9// [OpenGL ES 2.0.24] section 2.9 page 21.
10
daniel@transgaming.combbf56f72010-04-20 18:52:13 +000011#include "libGLESv2/Buffer.h"
daniel@transgaming.com4f39fd92010-03-08 20:26:45 +000012
daniel@transgaming.com0f7aaf52010-03-11 19:41:38 +000013#include <cstdlib>
14#include <limits>
15#include <utility>
16
alokp@chromium.orgea0e1af2010-03-22 19:33:14 +000017#include "common/debug.h"
daniel@transgaming.combbf56f72010-04-20 18:52:13 +000018
19#include "libGLESv2/geometry/backend.h"
daniel@transgaming.com4f39fd92010-03-08 20:26:45 +000020
21namespace gl
22{
daniel@transgaming.com4f39fd92010-03-08 20:26:45 +000023
daniel@transgaming.com0f7aaf52010-03-11 19:41:38 +000024Buffer::Buffer(BufferBackEnd *backEnd)
25 : mBackEnd(backEnd), mIdentityTranslation(NULL)
26{
daniel@transgaming.com4f39fd92010-03-08 20:26:45 +000027}
28
29Buffer::~Buffer()
30{
daniel@transgaming.com0f7aaf52010-03-11 19:41:38 +000031 delete mIdentityTranslation;
daniel@transgaming.com4f39fd92010-03-08 20:26:45 +000032}
33
daniel@transgaming.com0f7aaf52010-03-11 19:41:38 +000034GLenum Buffer::bufferData(const void* data, GLsizeiptr size, GLenum usage)
daniel@transgaming.com4f39fd92010-03-08 20:26:45 +000035{
daniel@transgaming.com0f7aaf52010-03-11 19:41:38 +000036 if (size < 0) return GL_INVALID_VALUE;
daniel@transgaming.com4f39fd92010-03-08 20:26:45 +000037
daniel@transgaming.comd4620a32010-03-21 04:31:28 +000038 const GLubyte* newdata = static_cast<const GLubyte*>(data);
daniel@transgaming.com4f39fd92010-03-08 20:26:45 +000039
daniel@transgaming.comc103b602010-04-23 18:34:58 +000040 if (size == 0)
41 {
42 mContents.clear();
43
44 delete mIdentityTranslation;
45 mIdentityTranslation = NULL;
46 }
47 else if (size != mContents.size())
daniel@transgaming.com4f39fd92010-03-08 20:26:45 +000048 {
daniel@transgaming.com0f7aaf52010-03-11 19:41:38 +000049 // vector::resize only provides the basic exception guarantee, so use temporaries & swap to get the strong exception guarantee.
50 // We don't want to risk having mContents and mIdentityTranslation that have different contents or even different sizes.
daniel@transgaming.comd4620a32010-03-21 04:31:28 +000051 std::vector<GLubyte> newContents;
52
53 if (newdata != NULL)
54 {
55 newContents.assign(newdata, newdata + size);
56 }
57 else
58 {
59 newContents.assign(size, 0);
60 }
daniel@transgaming.com0f7aaf52010-03-11 19:41:38 +000061
62 TranslatedVertexBuffer *newIdentityTranslation = mBackEnd->createVertexBuffer(size);
63
64 // No exceptions allowed after this point.
65
66 mContents.swap(newContents);
67
68 delete mIdentityTranslation;
69 mIdentityTranslation = newIdentityTranslation;
daniel@transgaming.com4f39fd92010-03-08 20:26:45 +000070 }
daniel@transgaming.comd4620a32010-03-21 04:31:28 +000071 else if (newdata != NULL)
daniel@transgaming.com0f7aaf52010-03-11 19:41:38 +000072 {
daniel@transgaming.com0f7aaf52010-03-11 19:41:38 +000073 mContents.assign(newdata, newdata + size);
74 }
75
daniel@transgaming.comaa0ccbd2010-04-15 20:45:05 +000076 mUsage = usage;
77
daniel@transgaming.com0f7aaf52010-03-11 19:41:38 +000078 return copyToIdentityBuffer(0, size);
daniel@transgaming.com4f39fd92010-03-08 20:26:45 +000079}
80
daniel@transgaming.com0f7aaf52010-03-11 19:41:38 +000081GLenum Buffer::bufferSubData(const void* data, GLsizeiptr size, GLintptr offset)
daniel@transgaming.com4f39fd92010-03-08 20:26:45 +000082{
daniel@transgaming.com0f7aaf52010-03-11 19:41:38 +000083 if (size < 0 || offset < 0) return GL_INVALID_VALUE;
84 if (std::numeric_limits<GLsizeiptr>::max() - offset < size) return GL_INVALID_VALUE;
85 if (size + offset > static_cast<GLsizeiptr>(mContents.size())) return GL_INVALID_VALUE;
daniel@transgaming.com4f39fd92010-03-08 20:26:45 +000086
daniel@transgaming.comd4620a32010-03-21 04:31:28 +000087 const GLubyte *newdata = static_cast<const GLubyte*>(data);
daniel@transgaming.com0f7aaf52010-03-11 19:41:38 +000088 copy(newdata, newdata + size, mContents.begin() + offset);
daniel@transgaming.com4f39fd92010-03-08 20:26:45 +000089
daniel@transgaming.com0f7aaf52010-03-11 19:41:38 +000090 return copyToIdentityBuffer(offset, size);
daniel@transgaming.com4f39fd92010-03-08 20:26:45 +000091}
92
daniel@transgaming.com0f7aaf52010-03-11 19:41:38 +000093GLenum Buffer::copyToIdentityBuffer(GLintptr offset, GLsizeiptr length)
daniel@transgaming.com4f39fd92010-03-08 20:26:45 +000094{
daniel@transgaming.com0f7aaf52010-03-11 19:41:38 +000095 ASSERT(offset >= 0 && length >= 0);
daniel@transgaming.com4f39fd92010-03-08 20:26:45 +000096
daniel@transgaming.comc103b602010-04-23 18:34:58 +000097 if (length > 0) // If length == 0 mIdentityTranslation might be NULL.
98 {
99 // This is a stalling map. Not great for performance.
100 GLubyte *p = static_cast<GLubyte*>(mIdentityTranslation->map());
daniel@transgaming.com4f39fd92010-03-08 20:26:45 +0000101
daniel@transgaming.comc103b602010-04-23 18:34:58 +0000102 memcpy(p + offset, &mContents[0] + offset, length);
103 mIdentityTranslation->unmap();
104 }
daniel@transgaming.com4f39fd92010-03-08 20:26:45 +0000105
daniel@transgaming.com0f7aaf52010-03-11 19:41:38 +0000106 return GL_NO_ERROR;
daniel@transgaming.com4f39fd92010-03-08 20:26:45 +0000107}
108
daniel@transgaming.com4f39fd92010-03-08 20:26:45 +0000109}