summaryrefslogtreecommitdiff
path: root/indra/llmath/llvolumemgr.h
blob: 2e0ce3e88ae1970f79bbbcb60c2db6ea2da1cdf8 (plain)
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
/**
 * @file llvolumemgr.h
 * @brief LLVolumeMgr class.
 *
 * $LicenseInfo:firstyear=2002&license=viewerlgpl$
 * Second Life Viewer Source Code
 * Copyright (C) 2010, Linden Research, Inc.
 *
 * This library is free software; you can redistribute it and/or
 * modify it under the terms of the GNU Lesser General Public
 * License as published by the Free Software Foundation;
 * version 2.1 of the License only.
 *
 * This library 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
 * Lesser General Public License for more details.
 *
 * You should have received a copy of the GNU Lesser General Public
 * License along with this library; if not, write to the Free Software
 * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA  02110-1301  USA
 *
 * Linden Research, Inc., 945 Battery Street, San Francisco, CA  94111  USA
 * $/LicenseInfo$
 */

#ifndef LL_LLVOLUMEMGR_H
#define LL_LLVOLUMEMGR_H

#include <map>

#include "llvolume.h"
#include "llpointer.h"
#include "llthread.h"

class LLVolumeParams;
class LLVolumeLODGroup;

class LLVolumeLODGroup
{
    LOG_CLASS(LLVolumeLODGroup);

public:
    enum
    {
        NUM_LODS = 4
    };

    LLVolumeLODGroup(const LLVolumeParams &params);
    ~LLVolumeLODGroup();
    bool cleanupRefs();

    static S32 getDetailFromTan(const F32 tan_angle);
    static void getDetailProximity(const F32 tan_angle, F32 &to_lower, F32& to_higher);
    static F32 getVolumeScaleFromDetail(const S32 detail);
    static S32 getVolumeDetailFromScale(F32 scale);

    LLVolume* refLOD(const S32 detail);
    bool derefLOD(LLVolume *volumep);
    S32 getNumRefs() const { return mRefs; }

    const LLVolumeParams* getVolumeParams() const { return &mVolumeParams; };

    F32 dump();
    friend std::ostream& operator<<(std::ostream& s, const LLVolumeLODGroup& volgroup);

protected:
    LLVolumeParams mVolumeParams;

    S32 mRefs;
    S32 mLODRefs[NUM_LODS];
    LLPointer<LLVolume> mVolumeLODs[NUM_LODS];
    static F32 mDetailThresholds[NUM_LODS];
    static F32 mDetailScales[NUM_LODS];
    S32     mAccessCount[NUM_LODS];
};

class LLVolumeMgr
{
public:
    LLVolumeMgr();
    virtual ~LLVolumeMgr();
    bool cleanup();         // Cleanup all volumes being managed, returns true if no dangling references

    virtual LLVolumeLODGroup* getGroup( const LLVolumeParams& volume_params ) const;

    // whatever calls getVolume() never owns the LLVolume* and
    // cannot keep references for long since it may be deleted
    // later.  For best results hold it in an LLPointer<LLVolume>.
    virtual LLVolume *refVolume(const LLVolumeParams &volume_params, const S32 detail);
    virtual void unrefVolume(LLVolume *volumep);

    void dump();

    // manually call this for mutex magic
    void useMutex();

    friend std::ostream& operator<<(std::ostream& s, const LLVolumeMgr& volume_mgr);

protected:
    void insertGroup(LLVolumeLODGroup* volgroup);
    // Overridden in llphysics/abstract/utils/llphysicsvolumemanager.h
    virtual LLVolumeLODGroup* createNewGroup(const LLVolumeParams& volume_params);

protected:
    typedef std::map<const LLVolumeParams*, LLVolumeLODGroup*, LLVolumeParams::compare> volume_lod_group_map_t;
    volume_lod_group_map_t mVolumeLODGroups;

    LLMutex* mDataMutex;
};

#endif // LL_LLVOLUMEMGR_H