C++程序  |  170行  |  5.28 KB

/*
 * Copyright 2014 Google Inc.
 *
 * Use of this source code is governed by a BSD-style license that can be
 * found in the LICENSE file.
 */

#if SK_SUPPORT_GPU

#include "gl/GrGLNameAllocator.h"
#include "Test.h"

////////////////////////////////////////////////////////////////////////////////

class NameLeakTest {
    static const GrGLuint kFirstName = 101;
    static const GrGLuint kRange = 1013;

public:
    NameLeakTest(skiatest::Reporter* reporter)
        : fReporter(reporter),
          fAllocator(kFirstName, kFirstName + kRange),
          fAllocatedCount(0),
          fRandomName(kFirstName + 4 * kRange / 7) {
        memset(fAllocatedNames, 0, sizeof(fAllocatedNames));
    }

    bool run() {
        if (!this->allocateAllRemaining()) {
            return false;
        }

        for (GrGLuint freeCount = 1; freeCount <= kRange; ++freeCount) {
            if (!this->freeRandomNames(freeCount)) {
                return false;
            }
            if (!this->allocateAllRemaining()) {
                return false;
            }
        }

        return true;
    }

private:
    bool isAllocated(GrGLuint name) const {
        return fAllocatedNames[name - kFirstName];
    }

    void setAllocated(GrGLuint name, bool allocated) {
        fAllocatedNames[name - kFirstName] = allocated;
    }

    bool allocateAllRemaining() {
        for (; fAllocatedCount < kRange; ++fAllocatedCount) {
            GrGLuint name = fAllocator.allocateName();
            if (0 == name) {
                ERRORF(fReporter,
                       "Name allocate failed, but there should still be %u free names",
                       kRange - fAllocatedCount);
                return false;
            }
            if (name < kFirstName || name >= kFirstName + kRange) {
                ERRORF(fReporter,
                       "Name allocate returned name %u outside its bounds [%u, %u)",
                       name, kFirstName, kFirstName + kRange);
                return false;
            }
            if (this->isAllocated(name)) {
                ERRORF(fReporter, "Name allocate returned name that is already allocated");
                return false;
            }

            this->setAllocated(name, true);
        }

        // Ensure it returns 0 once all the names are allocated.
        GrGLuint name = fAllocator.allocateName();
        if (0 != name) {
            ERRORF(fReporter,
                   "Name allocate did not fail when all names were already in use");
            return false;
        }

        // Ensure every unique name is allocated.
        for (GrGLuint i = 0; i < kRange; ++i) {
            if (!this->isAllocated(kFirstName + i)) {
                ERRORF(fReporter, "Not all unique names are allocated after allocateAllRemaining()");
                return false;
            }
        }

        return true;
    }

    bool freeRandomNames(GrGLuint count) {
        // The values a and c make up an LCG (pseudo-random generator). These
        // values must satisfy the Hull-Dobell Theorem (with m=kRange):
        // http://en.wikipedia.org/wiki/Linear_congruential_generator
        // We use our own generator to guarantee it hits each unique value
        // within kRange exactly once before repeating.
        const GrGLuint seed = (count + fRandomName) / 2;
        const GrGLuint a = seed * kRange + 1;
        const GrGLuint c = (seed * 743) % kRange;

        for (GrGLuint i = 0; i < count; ++i) {
            fRandomName = (a * fRandomName + c) % kRange;
            const GrGLuint name = kFirstName + fRandomName;
            if (!this->isAllocated(name)) {
                ERRORF(fReporter, "Test bug: Should not free a not-allocated name at this point (%u)", i);
                return false;
            }

            fAllocator.free(name);
            this->setAllocated(name, false);
            --fAllocatedCount;
        }

        return true;
    }

    skiatest::Reporter* fReporter;
    GrGLNameAllocator fAllocator;
    bool fAllocatedNames[kRange];
    GrGLuint fAllocatedCount;
    GrGLuint fRandomName;
};

DEF_GPUTEST(NameAllocator, reporter, factory) {
    // Ensure no names are leaked or double-allocated during heavy usage.
    {
        NameLeakTest nameLeakTest(reporter);
        nameLeakTest.run();
    }

    static const GrGLuint range = 32;
    GrGLNameAllocator allocator(1, 1 + range);
    for (GrGLuint i = 1; i <= range; ++i) {
        allocator.allocateName();
    }
    REPORTER_ASSERT(reporter, 0 == allocator.allocateName());

    // Test freeing names out of range.
    allocator.free(allocator.firstName() - 1);
    allocator.free(allocator.endName());
    REPORTER_ASSERT(reporter, 0 == allocator.allocateName());

    // Test freeing not-allocated names.
    for (GrGLuint i = 1; i <= range/2; i += 2) {
        allocator.free(i);
    }
    for (GrGLuint i = 1; i <= range/2; i += 2) {
        // None of these names will be allocated.
        allocator.free(i);
    }
    for (GrGLuint i = 1; i <= range/2; ++i) {
        // Every other name will not be be allocated.
        allocator.free(i);
    }
    for (GrGLuint i = 1; i <= range/2; ++i) {
        if (0 == allocator.allocateName()) {
            ERRORF(reporter, "Name allocate failed when there should be free names");
            break;
        }
    }
    REPORTER_ASSERT(reporter, 0 == allocator.allocateName());
}

#endif