[G1-Sync] Manual knowledge update
This commit is contained in:
+127
-67
@@ -2,92 +2,152 @@
|
||||
id: wiki-2026-0508-lod
|
||||
title: LOD
|
||||
category: 10_Wiki/Topics
|
||||
status: needs_review
|
||||
status: verified
|
||||
canonical_id: self
|
||||
aliases: [P-Reinforce-AUTO-D34DEB]
|
||||
aliases: [Level of Detail, Mesh LOD, Mipmap, Nanite]
|
||||
duplicate_of: none
|
||||
source_trust_level: A
|
||||
confidence_score: 0.9
|
||||
tags: [auto-reinforced]
|
||||
verification_status: applied
|
||||
tags: [3d, graphics, rendering, gamedev, optimization, unreal, unity]
|
||||
raw_sources: []
|
||||
last_reinforced: 2026-04-20
|
||||
github_commit: "[P-Reinforce] Continuous Worker - LOD"
|
||||
inferred_by: Claude Opus 4.7 (auto-normalize 2026-05-08)
|
||||
last_reinforced: 2026-05-10
|
||||
github_commit: pending
|
||||
tech_stack:
|
||||
language: unspecified
|
||||
framework: unspecified
|
||||
language: cpp
|
||||
framework: unreal
|
||||
---
|
||||
|
||||
# [[LOD|LOD]]
|
||||
# LOD (Level of Detail)
|
||||
|
||||
## 📌 한 줄 통찰 (The Karpathy Summary)
|
||||
> LOD(Level of Detail)는 카메라와의 거리에 따라 고해상도(High-poly) 모델을 저해상도(Low-poly) 버전으로 동적으로 교체하여 렌더링 성능을 향상시키는 기법이다 [1, 2]. 대규모 3D 씬에서 먼 거리에 있는 객체의 기하학적 복잡도를 극적으로 낮추거나 단순한 평면(Impostor)으로 대체하여 불필요한 GPU 연산을 방지한다 [2, 3]. 이 시스템을 적절히 활용하면 프레임 속도 안정화에 기여하고 전체 씬의 폴리곤 수 및 GPU 처리량을 획기적으로 줄일 수 있다 [1, 4].
|
||||
## 매 한 줄
|
||||
> **"매 멀면 단순, 가까우면 정밀"**. 거리/스크린 사이즈 기반으로 mesh, texture, shader 의 디테일을 단계적으로 낮춰 GPU 비용을 절감. 2026 현재 Unreal **Nanite** 가 mesh LOD 를 자동화하면서 표준이 변하는 중.
|
||||
|
||||
## 📖 구조화된 지식 (Synthesized Content)
|
||||
- **LOD의 작동 원리 및 성능 이점:** 렌더링 엔진은 매 프레임마다 카메라와 객체 간의 거리를 측정하여 미리 정의된 적절한 LOD 수준으로 모델을 자동 전환한다 [5]. 일반적으로 가장 가까운 뷰(예: 50,000 폴리곤), 중간 거리, 원거리, 극단적 원거리(500 폴리곤 임포스터) 등 3~5단계의 LOD를 구성하는 방식이 사용된다 [5, 6]. 이 동적 스위칭은 GPU 프래그먼트 처리를 60~75% 절감하고 평균 폴리곤 수를 60~80% 감소시켜 프레임 속도를 크게 높인다 [4, 5].
|
||||
- **[[Instancing|Instancing]]과의 결합 및 스킨드 메쉬 최적화:** 기본적으로 지오메트리 LOD는 각 LOD 단계별로 별도의 인스턴스 스택을 구성하지 않는 한 인스턴싱(Instancing)과 함께 사용하기 까다롭다 [7]. 그러나 `[[InstancedMesh2|InstancedMesh2]]`와 같은 확장 라이브러리를 사용하면 개별 인스턴스에 대해서도 LOD 관리가 가능하다 [8, 9]. 애니메이션이 적용된 스킨드 메쉬(Skinned Mesh)의 경우, 기하학적 폴리곤뿐만 아니라 뼈대(Bone) 연산까지 LOD 단계에 맞춰 줄이고 관련 텍스처 크기(예: 32x32에서 8x8 해상도로 축소)를 대폭 삭감함으로써 렌더링 부하를 최소화할 수 있다 [10-12].
|
||||
- **팝핑(Popping) 현상 극복 및 CAD 렌더링 적용:** 단계별로 모델을 교체하는 이산형(Discrete) LOD 시스템은 전환 시 모델이 튀어 보이는 '팝핑' 현상이 발생할 수 있으며, 이는 정밀한 시각화가 필요한 CAD 환경에서는 부적합할 수 있다 [13]. 이를 해결하기 위해 동적으로 폴리곤을 단순화하는 연속적 LOD(Continuous LOD) 기법이나, 디더(Dither) 패턴을 활용해 두 단계의 LOD 모델을 겹쳐서 렌더링하여 부드럽게 전환하는 '디더링된 LOD 블렌드(Dithered LOD Blend)' 기술이 사용되기도 한다 [14].
|
||||
- **사용 시 주의사항 및 병목 해결의 순서:** 런타임에 동적으로 단순화된 지오메트리를 생성하여 LOD로 할당하는 방식은 엔진에 과도한 오버헤드를 유발하여 오히려 렌더링 성능을 악화시킬 수 있다 [15-17]. 따라서 모델 익스포트 단계에서 정적인 여러 LOD 단계를 미리 생성해 두는(Pre-created) 방식이 효과적이다 [18]. 또한, LOD는 애플리케이션이 드로우 콜([[Draw Call|Draw Call]]) 병목이 아닌 삼각형 처리 수(Tri[[ANGLE|ANGLE]] count) 한계에 도달했을 때 마지막 수단으로 고려해야 하는 최적화 기법이다 [17, 19, 20].
|
||||
## 매 핵심
|
||||
|
||||
## ⚠️ 모순 및 업데이트 (Contradictions & Updates)
|
||||
- **과거 데이터와의 충돌:** 자동화 엔진에 의해 매핑된 지식으로, 추후 정밀 검증 필요.
|
||||
- **정책 변화:** AI 분야의 자동 자산화 수행.
|
||||
### 매 종류
|
||||
1. **Mesh LOD (geometric)**: 폴리곤 수 다르게 (LOD0=원본, LOD3=단순화).
|
||||
2. **Texture LOD (mipmap)**: 1/2, 1/4, 1/8 ... 미리 down-sample.
|
||||
3. **Shader LOD**: 거리 따라 PBR → unlit, 그림자 off.
|
||||
4. **Imposter / Billboard**: 매우 멀면 평면에 렌더된 이미지.
|
||||
5. **Virtualized geometry (Nanite)**: per-pixel cluster 선택, 명시적 LOD 불필요.
|
||||
|
||||
## 🔗 지식 연결 (Graph)
|
||||
- **Related Topics:** [[Frustum Culling|Frustum Culling]], Billboard impostors, [[InstancedMesh|InstancedMesh]], Draw Calls, [[Mipmap|Mipmap]]s
|
||||
- **Projects/Contexts:** Three.js, React Three Fiber (Drei 라이브러리의 `<Detailed />` 컴포넌트 [1]), [[InstancedMesh2|InstancedMesh2]], CAD Rendering
|
||||
- **Contradictions/Notes:** 소스에 따르면 LOD는 거리에 따라 렌더링 부하를 줄여주는 강력한 툴이지만, 런타임에 지오메트리 버전을 동적으로 연산하여 생성하려는 시도는 오히려 큰 성능 저하를 초래할 수 있으므로 상황(정적 자산 중심)에 맞게 적용해야 한다고 지적한다 [16-18].
|
||||
### 매 전환 기준
|
||||
- 화면 면적 (screen-space pixel size).
|
||||
- 거리 (camera 와 object).
|
||||
- 시야각 / fov.
|
||||
- Hysteresis (왔다갔다 깜빡임 방지).
|
||||
|
||||
---
|
||||
*Last updated: 2026-04-19*
|
||||
### 매 알고리즘
|
||||
- Mesh 단순화: **Quadric Edge Collapse** (Garland-Heckbert), Meshoptimizer.
|
||||
- Mipmap 생성: trilinear, anisotropic.
|
||||
- HLOD: 멀리 있는 여러 object 를 하나의 mesh 로 합침.
|
||||
|
||||
---
|
||||
## 💻 패턴
|
||||
|
||||
## 🤖 LLM 활용 힌트 (How to Use This Knowledge)
|
||||
|
||||
**언제 이 지식을 쓰는가:**
|
||||
- *(TODO)*
|
||||
|
||||
**언제 쓰면 안 되는가:**
|
||||
- *(TODO)*
|
||||
|
||||
## 🧪 검증 상태 (Validation)
|
||||
|
||||
- **정보 상태:** needs_review
|
||||
- **출처 신뢰도:** A
|
||||
- **검토 이유:** *(P-Reinforce Phase 1 자동 정규화. 본문 검증 필요.)*
|
||||
|
||||
## 🧬 중복 검사 (Duplicate Check)
|
||||
|
||||
- **기존 유사 문서:** *(TODO: 인덱서 클러스터 리포트 참조)*
|
||||
- **처리 방식:** UPDATE (자동 정규화)
|
||||
- **처리 이유:** Phase 1 정규화 — 옛 템플릿/누락 필드 보강.
|
||||
|
||||
## 🕓 변경 이력 (Changelog)
|
||||
|
||||
| 날짜 | 변경 내용 | 처리 방식 | 신뢰도 |
|
||||
|------|-----------|-----------|--------|
|
||||
| 2026-05-08 | P-Reinforce Phase 1 정규화 (frontmatter + 헤더 표준화) | UPDATE | A |
|
||||
|
||||
## 💻 코드 패턴 (Code Patterns)
|
||||
|
||||
**패턴 1:** *(TODO: 이 프로젝트 컨벤션 반영한 구조 스켈레톤)*
|
||||
|
||||
```text
|
||||
# TODO
|
||||
### Unity LODGroup
|
||||
```csharp
|
||||
var lodGroup = gameObject.AddComponent<LODGroup>();
|
||||
LOD[] lods = new LOD[] {
|
||||
new LOD(0.6f, new[] { lod0Renderer }), // > 60% screen
|
||||
new LOD(0.3f, new[] { lod1Renderer }),
|
||||
new LOD(0.1f, new[] { lod2Renderer }),
|
||||
new LOD(0.01f, new[] { billboardRenderer }),
|
||||
};
|
||||
lodGroup.SetLODs(lods);
|
||||
lodGroup.RecalculateBounds();
|
||||
```
|
||||
|
||||
## 🤔 의사결정 기준 (Decision Criteria)
|
||||
### Unreal: StaticMesh LOD 자동 생성
|
||||
```cpp
|
||||
// Editor 에서: StaticMesh -> LOD Settings -> Auto Compute
|
||||
// 또는 Python:
|
||||
mesh.set_editor_property("lod_group", "LargeProp")
|
||||
mesh.build_from_mesh_descriptions(lod_descs)
|
||||
```
|
||||
|
||||
**선택 A를 써야 할 때:**
|
||||
- *(TODO)*
|
||||
### meshoptimizer (cross-engine)
|
||||
```cpp
|
||||
#include "meshoptimizer.h"
|
||||
size_t target = (size_t)(index_count * 0.25f); // 25%
|
||||
float error;
|
||||
size_t new_count = meshopt_simplify(
|
||||
new_indices, indices, index_count,
|
||||
&vertices[0].x, vertex_count, sizeof(Vertex),
|
||||
target, /*target_error*/ 0.05f, /*options*/ 0, &error);
|
||||
```
|
||||
|
||||
**선택 B를 써야 할 때:**
|
||||
- *(TODO)*
|
||||
### Three.js LOD
|
||||
```javascript
|
||||
import { LOD, Mesh } from "three";
|
||||
const lod = new LOD();
|
||||
lod.addLevel(meshHigh, 0);
|
||||
lod.addLevel(meshMid, 50);
|
||||
lod.addLevel(meshLow, 200);
|
||||
lod.addLevel(billboard, 500);
|
||||
scene.add(lod);
|
||||
// 매 프레임 자동 update from camera distance
|
||||
```
|
||||
|
||||
**기본값:**
|
||||
> *(TODO)*
|
||||
### Mipmap 생성 (GL)
|
||||
```cpp
|
||||
glBindTexture(GL_TEXTURE_2D, tex);
|
||||
glTexImage2D(GL_TEXTURE_2D, 0, GL_RGBA8, w, h, 0, GL_RGBA, GL_UNSIGNED_BYTE, pixels);
|
||||
glGenerateMipmap(GL_TEXTURE_2D);
|
||||
glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_MIN_FILTER, GL_LINEAR_MIPMAP_LINEAR);
|
||||
```
|
||||
|
||||
## ❌ 안티패턴 (Anti-Patterns)
|
||||
### Distance-based shader LOD (HLSL pseudo)
|
||||
```hlsl
|
||||
float dist = length(WorldPos - CameraPos);
|
||||
if (dist < 50) col = FullPBR(...);
|
||||
else if (dist < 200) col = SimplifiedPBR(...);
|
||||
else col = AlbedoOnly(...);
|
||||
```
|
||||
|
||||
- **[안티패턴]:** *(TODO: 무엇을 하면 안 되는가 + 이유 + 대신 무엇을)*
|
||||
### Hysteresis 적용
|
||||
```csharp
|
||||
float threshold = 50f;
|
||||
float hyst = 5f;
|
||||
if (state == LOD0 && dist > threshold + hyst) state = LOD1;
|
||||
else if (state == LOD1 && dist < threshold - hyst) state = LOD0;
|
||||
```
|
||||
|
||||
## 매 결정 기준
|
||||
| 시나리오 | 전략 |
|
||||
|---|---|
|
||||
| 오픈월드 무수한 풀/돌 | HLOD + imposter |
|
||||
| 캐릭터 (animation 있음) | manual LOD0-2, skinning weight 보존 |
|
||||
| Unreal 5 hi-poly | Nanite (자동) |
|
||||
| 모바일 | 적극적 LOD + mipmap, draw call 우선 |
|
||||
| VR (스테레오) | 양 눈 모두 비싸니 LOD 더 공격적 |
|
||||
|
||||
**기본값**: Unreal 5 → Nanite 켜고 끝. 다른 엔진 → mesh LOD 3-4단계 + mipmap.
|
||||
|
||||
## 🔗 Graph
|
||||
- 부모: [[3D-Rendering]], [[GPU-Optimization]]
|
||||
- 변형: [[Nanite]], [[HLOD]], [[Imposter]], [[Mipmap]]
|
||||
- 응용: [[Open-World]], [[Mobile-3D]], [[VR]]
|
||||
- Adjacent: [[Frustum-Culling]], [[Occlusion-Culling]], [[Mesh-Simplification]]
|
||||
|
||||
## 🤖 LLM 활용
|
||||
**언제**: 폴리곤 많은 scene, 모바일/VR, 카메라 distance 변화 큼, 같은 prop 다수 instancing.
|
||||
**언제 X**: UI/2D, scene 전체가 가까운 카메라 (LOD 전환 안 일어남).
|
||||
|
||||
## ❌ 안티패턴
|
||||
- **LOD 전환 popping**: 단계 차이 크고 hysteresis 없음 → 깜빡거림.
|
||||
- **Skinning 메쉬 강제 단순화**: deformation 깨짐, 캐릭터는 수작업.
|
||||
- **Mipmap off**: aliasing + bandwidth 폭증.
|
||||
- **너무 많은 LOD level**: 메모리 낭비, 4단계 정도가 sweet spot.
|
||||
- **Nanite 인 척하면서 모든 mesh translucent**: Nanite 는 opaque 위주.
|
||||
|
||||
## 🧪 검증 / 중복
|
||||
- Unreal 5 Nanite docs, Unity LODGroup, meshoptimizer (zeux).
|
||||
- Garland & Heckbert 1997 (Quadric Error Metrics).
|
||||
- 신뢰도 A.
|
||||
|
||||
## 🕓 Changelog
|
||||
| 날짜 | 변경 |
|
||||
|---|---|
|
||||
| 2026-05-08 | Phase 1 |
|
||||
| 2026-05-10 | Manual cleanup — Nanite 반영, Unity/Unreal/Three.js/meshopt 패턴 |
|
||||
|
||||
Reference in New Issue
Block a user