diff options
Diffstat (limited to 'OpenSim/Region/Framework/Scenes/Scene.cs')
-rw-r--r-- | OpenSim/Region/Framework/Scenes/Scene.cs | 307 |
1 files changed, 153 insertions, 154 deletions
diff --git a/OpenSim/Region/Framework/Scenes/Scene.cs b/OpenSim/Region/Framework/Scenes/Scene.cs index 02a0268..e0af2d6 100644 --- a/OpenSim/Region/Framework/Scenes/Scene.cs +++ b/OpenSim/Region/Framework/Scenes/Scene.cs | |||
@@ -129,7 +129,16 @@ namespace OpenSim.Region.Framework.Scenes | |||
129 | protected ICapabilitiesModule m_capsModule; | 129 | protected ICapabilitiesModule m_capsModule; |
130 | // Central Update Loop | 130 | // Central Update Loop |
131 | protected int m_fps = 10; | 131 | protected int m_fps = 10; |
132 | protected uint m_frame; | 132 | |
133 | /// <summary> | ||
134 | /// Current scene frame number | ||
135 | /// </summary> | ||
136 | public uint Frame | ||
137 | { | ||
138 | get; | ||
139 | protected set; | ||
140 | } | ||
141 | |||
133 | protected float m_timespan = 0.089f; | 142 | protected float m_timespan = 0.089f; |
134 | protected DateTime m_lastupdate = DateTime.UtcNow; | 143 | protected DateTime m_lastupdate = DateTime.UtcNow; |
135 | 144 | ||
@@ -1183,7 +1192,8 @@ namespace OpenSim.Region.Framework.Scenes | |||
1183 | 1192 | ||
1184 | try | 1193 | try |
1185 | { | 1194 | { |
1186 | Update(); | 1195 | while (!shuttingdown) |
1196 | Update(); | ||
1187 | 1197 | ||
1188 | m_lastUpdate = Util.EnvironmentTickCount(); | 1198 | m_lastUpdate = Util.EnvironmentTickCount(); |
1189 | m_firstHeartbeat = false; | 1199 | m_firstHeartbeat = false; |
@@ -1200,187 +1210,176 @@ namespace OpenSim.Region.Framework.Scenes | |||
1200 | Watchdog.RemoveThread(); | 1210 | Watchdog.RemoveThread(); |
1201 | } | 1211 | } |
1202 | 1212 | ||
1203 | /// <summary> | ||
1204 | /// Performs per-frame updates on the scene, this should be the central scene loop | ||
1205 | /// </summary> | ||
1206 | public override void Update() | 1213 | public override void Update() |
1207 | { | 1214 | { |
1208 | float physicsFPS; | 1215 | TimeSpan SinceLastFrame = DateTime.UtcNow - m_lastupdate; |
1209 | int maintc; | 1216 | float physicsFPS = 0f; |
1210 | 1217 | ||
1211 | while (!shuttingdown) | 1218 | int maintc = Util.EnvironmentTickCount(); |
1219 | int tmpFrameMS = maintc; | ||
1220 | tempOnRezMS = eventMS = backupMS = terrainMS = landMS = 0; | ||
1221 | |||
1222 | // Increment the frame counter | ||
1223 | ++Frame; | ||
1224 | |||
1225 | try | ||
1212 | { | 1226 | { |
1213 | TimeSpan SinceLastFrame = DateTime.UtcNow - m_lastupdate; | 1227 | // Check if any objects have reached their targets |
1214 | physicsFPS = 0f; | 1228 | CheckAtTargets(); |
1215 | 1229 | ||
1216 | maintc = Util.EnvironmentTickCount(); | 1230 | // Update SceneObjectGroups that have scheduled themselves for updates |
1217 | int tmpFrameMS = maintc; | 1231 | // Objects queue their updates onto all scene presences |
1218 | tempOnRezMS = eventMS = backupMS = terrainMS = landMS = 0; | 1232 | if (Frame % m_update_objects == 0) |
1233 | m_sceneGraph.UpdateObjectGroups(); | ||
1219 | 1234 | ||
1220 | // Increment the frame counter | 1235 | // Run through all ScenePresences looking for updates |
1221 | ++m_frame; | 1236 | // Presence updates and queued object updates for each presence are sent to clients |
1237 | if (Frame % m_update_presences == 0) | ||
1238 | m_sceneGraph.UpdatePresences(); | ||
1222 | 1239 | ||
1223 | try | 1240 | // Coarse locations relate to positions of green dots on the mini-map (on a SecondLife client) |
1241 | if (Frame % m_update_coarse_locations == 0) | ||
1224 | { | 1242 | { |
1225 | // Check if any objects have reached their targets | 1243 | List<Vector3> coarseLocations; |
1226 | CheckAtTargets(); | 1244 | List<UUID> avatarUUIDs; |
1227 | 1245 | SceneGraph.GetCoarseLocations(out coarseLocations, out avatarUUIDs, 60); | |
1228 | // Update SceneObjectGroups that have scheduled themselves for updates | 1246 | // Send coarse locations to clients |
1229 | // Objects queue their updates onto all scene presences | 1247 | ForEachScenePresence(delegate(ScenePresence presence) |
1230 | if (m_frame % m_update_objects == 0) | 1248 | { |
1231 | m_sceneGraph.UpdateObjectGroups(); | 1249 | presence.SendCoarseLocations(coarseLocations, avatarUUIDs); |
1250 | }); | ||
1251 | } | ||
1232 | 1252 | ||
1233 | // Run through all ScenePresences looking for updates | 1253 | int tmpPhysicsMS2 = Util.EnvironmentTickCount(); |
1234 | // Presence updates and queued object updates for each presence are sent to clients | 1254 | if ((Frame % m_update_physics == 0) && m_physics_enabled) |
1235 | if (m_frame % m_update_presences == 0) | 1255 | m_sceneGraph.UpdatePreparePhysics(); |
1236 | m_sceneGraph.UpdatePresences(); | 1256 | physicsMS2 = Util.EnvironmentTickCountSubtract(tmpPhysicsMS2); |
1237 | 1257 | ||
1238 | // Coarse locations relate to positions of green dots on the mini-map (on a SecondLife client) | 1258 | // Apply any pending avatar force input to the avatar's velocity |
1239 | if (m_frame % m_update_coarse_locations == 0) | 1259 | if (Frame % m_update_entitymovement == 0) |
1240 | { | 1260 | m_sceneGraph.UpdateScenePresenceMovement(); |
1241 | List<Vector3> coarseLocations; | ||
1242 | List<UUID> avatarUUIDs; | ||
1243 | SceneGraph.GetCoarseLocations(out coarseLocations, out avatarUUIDs, 60); | ||
1244 | // Send coarse locations to clients | ||
1245 | ForEachScenePresence(delegate(ScenePresence presence) | ||
1246 | { | ||
1247 | presence.SendCoarseLocations(coarseLocations, avatarUUIDs); | ||
1248 | }); | ||
1249 | } | ||
1250 | 1261 | ||
1251 | int tmpPhysicsMS2 = Util.EnvironmentTickCount(); | 1262 | // Perform the main physics update. This will do the actual work of moving objects and avatars according to their |
1252 | if ((m_frame % m_update_physics == 0) && m_physics_enabled) | 1263 | // velocity |
1253 | m_sceneGraph.UpdatePreparePhysics(); | 1264 | int tmpPhysicsMS = Util.EnvironmentTickCount(); |
1254 | physicsMS2 = Util.EnvironmentTickCountSubtract(tmpPhysicsMS2); | 1265 | if (Frame % m_update_physics == 0) |
1266 | { | ||
1267 | if (m_physics_enabled) | ||
1268 | physicsFPS = m_sceneGraph.UpdatePhysics(Math.Max(SinceLastFrame.TotalSeconds, m_timespan)); | ||
1269 | if (SynchronizeScene != null) | ||
1270 | SynchronizeScene(this); | ||
1271 | } | ||
1272 | physicsMS = Util.EnvironmentTickCountSubtract(tmpPhysicsMS); | ||
1255 | 1273 | ||
1256 | // Apply any pending avatar force input to the avatar's velocity | 1274 | // Delete temp-on-rez stuff |
1257 | if (m_frame % m_update_entitymovement == 0) | 1275 | if (Frame % 1000 == 0 && !m_cleaningTemps) |
1258 | m_sceneGraph.UpdateScenePresenceMovement(); | 1276 | { |
1277 | int tmpTempOnRezMS = Util.EnvironmentTickCount(); | ||
1278 | m_cleaningTemps = true; | ||
1279 | Util.FireAndForget(delegate { CleanTempObjects(); m_cleaningTemps = false; }); | ||
1280 | tempOnRezMS = Util.EnvironmentTickCountSubtract(tmpTempOnRezMS); | ||
1281 | } | ||
1259 | 1282 | ||
1260 | // Perform the main physics update. This will do the actual work of moving objects and avatars according to their | 1283 | if (RegionStatus != RegionStatus.SlaveScene) |
1261 | // velocity | 1284 | { |
1262 | int tmpPhysicsMS = Util.EnvironmentTickCount(); | 1285 | if (Frame % m_update_events == 0) |
1263 | if (m_frame % m_update_physics == 0) | ||
1264 | { | 1286 | { |
1265 | if (m_physics_enabled) | 1287 | int evMS = Util.EnvironmentTickCount(); |
1266 | physicsFPS = m_sceneGraph.UpdatePhysics(Math.Max(SinceLastFrame.TotalSeconds, m_timespan)); | 1288 | UpdateEvents(); |
1267 | if (SynchronizeScene != null) | 1289 | eventMS = Util.EnvironmentTickCountSubtract(evMS); ; |
1268 | SynchronizeScene(this); | ||
1269 | } | 1290 | } |
1270 | physicsMS = Util.EnvironmentTickCountSubtract(tmpPhysicsMS); | ||
1271 | 1291 | ||
1272 | // Delete temp-on-rez stuff | 1292 | if (Frame % m_update_backup == 0) |
1273 | if (m_frame % 1000 == 0 && !m_cleaningTemps) | ||
1274 | { | 1293 | { |
1275 | int tmpTempOnRezMS = Util.EnvironmentTickCount(); | 1294 | int backMS = Util.EnvironmentTickCount(); |
1276 | m_cleaningTemps = true; | 1295 | UpdateStorageBackup(); |
1277 | Util.FireAndForget(delegate { CleanTempObjects(); m_cleaningTemps = false; }); | 1296 | backupMS = Util.EnvironmentTickCountSubtract(backMS); |
1278 | tempOnRezMS = Util.EnvironmentTickCountSubtract(tmpTempOnRezMS); | ||
1279 | } | 1297 | } |
1280 | 1298 | ||
1281 | if (RegionStatus != RegionStatus.SlaveScene) | 1299 | if (Frame % m_update_terrain == 0) |
1282 | { | 1300 | { |
1283 | if (m_frame % m_update_events == 0) | 1301 | int terMS = Util.EnvironmentTickCount(); |
1284 | { | 1302 | UpdateTerrain(); |
1285 | int evMS = Util.EnvironmentTickCount(); | 1303 | terrainMS = Util.EnvironmentTickCountSubtract(terMS); |
1286 | UpdateEvents(); | 1304 | } |
1287 | eventMS = Util.EnvironmentTickCountSubtract(evMS); ; | ||
1288 | } | ||
1289 | |||
1290 | if (m_frame % m_update_backup == 0) | ||
1291 | { | ||
1292 | int backMS = Util.EnvironmentTickCount(); | ||
1293 | UpdateStorageBackup(); | ||
1294 | backupMS = Util.EnvironmentTickCountSubtract(backMS); | ||
1295 | } | ||
1296 | 1305 | ||
1297 | if (m_frame % m_update_terrain == 0) | 1306 | //if (Frame % m_update_land == 0) |
1298 | { | 1307 | //{ |
1299 | int terMS = Util.EnvironmentTickCount(); | 1308 | // int ldMS = Util.EnvironmentTickCount(); |
1300 | UpdateTerrain(); | 1309 | // UpdateLand(); |
1301 | terrainMS = Util.EnvironmentTickCountSubtract(terMS); | 1310 | // landMS = Util.EnvironmentTickCountSubtract(ldMS); |
1302 | } | 1311 | //} |
1303 | 1312 | ||
1304 | //if (m_frame % m_update_land == 0) | 1313 | frameMS = Util.EnvironmentTickCountSubtract(tmpFrameMS); |
1305 | //{ | 1314 | otherMS = tempOnRezMS + eventMS + backupMS + terrainMS + landMS; |
1306 | // int ldMS = Util.EnvironmentTickCount(); | 1315 | lastCompletedFrame = Util.EnvironmentTickCount(); |
1307 | // UpdateLand(); | 1316 | |
1308 | // landMS = Util.EnvironmentTickCountSubtract(ldMS); | 1317 | // if (Frame%m_update_avatars == 0) |
1309 | //} | 1318 | // UpdateInWorldTime(); |
1319 | StatsReporter.AddPhysicsFPS(physicsFPS); | ||
1320 | StatsReporter.AddTimeDilation(TimeDilation); | ||
1321 | StatsReporter.AddFPS(1); | ||
1322 | StatsReporter.SetRootAgents(m_sceneGraph.GetRootAgentCount()); | ||
1323 | StatsReporter.SetChildAgents(m_sceneGraph.GetChildAgentCount()); | ||
1324 | StatsReporter.SetObjects(m_sceneGraph.GetTotalObjectsCount()); | ||
1325 | StatsReporter.SetActiveObjects(m_sceneGraph.GetActiveObjectsCount()); | ||
1326 | StatsReporter.addFrameMS(frameMS); | ||
1327 | StatsReporter.addPhysicsMS(physicsMS + physicsMS2); | ||
1328 | StatsReporter.addOtherMS(otherMS); | ||
1329 | StatsReporter.SetActiveScripts(m_sceneGraph.GetActiveScriptsCount()); | ||
1330 | StatsReporter.addScriptLines(m_sceneGraph.GetScriptLPS()); | ||
1331 | } | ||
1310 | 1332 | ||
1311 | frameMS = Util.EnvironmentTickCountSubtract(tmpFrameMS); | 1333 | if (LoginsDisabled && Frame == 20) |
1312 | otherMS = tempOnRezMS + eventMS + backupMS + terrainMS + landMS; | 1334 | { |
1313 | lastCompletedFrame = Util.EnvironmentTickCount(); | 1335 | // In 99.9% of cases it is a bad idea to manually force garbage collection. However, |
1314 | 1336 | // this is a rare case where we know we have just went through a long cycle of heap | |
1315 | // if (m_frame%m_update_avatars == 0) | 1337 | // allocations, and there is no more work to be done until someone logs in |
1316 | // UpdateInWorldTime(); | 1338 | GC.Collect(); |
1317 | StatsReporter.AddPhysicsFPS(physicsFPS); | ||
1318 | StatsReporter.AddTimeDilation(TimeDilation); | ||
1319 | StatsReporter.AddFPS(1); | ||
1320 | StatsReporter.SetRootAgents(m_sceneGraph.GetRootAgentCount()); | ||
1321 | StatsReporter.SetChildAgents(m_sceneGraph.GetChildAgentCount()); | ||
1322 | StatsReporter.SetObjects(m_sceneGraph.GetTotalObjectsCount()); | ||
1323 | StatsReporter.SetActiveObjects(m_sceneGraph.GetActiveObjectsCount()); | ||
1324 | StatsReporter.addFrameMS(frameMS); | ||
1325 | StatsReporter.addPhysicsMS(physicsMS + physicsMS2); | ||
1326 | StatsReporter.addOtherMS(otherMS); | ||
1327 | StatsReporter.SetActiveScripts(m_sceneGraph.GetActiveScriptsCount()); | ||
1328 | StatsReporter.addScriptLines(m_sceneGraph.GetScriptLPS()); | ||
1329 | } | ||
1330 | 1339 | ||
1331 | if (LoginsDisabled && m_frame == 20) | 1340 | IConfig startupConfig = m_config.Configs["Startup"]; |
1341 | if (startupConfig == null || !startupConfig.GetBoolean("StartDisabled", false)) | ||
1332 | { | 1342 | { |
1333 | // In 99.9% of cases it is a bad idea to manually force garbage collection. However, | 1343 | m_log.DebugFormat("[REGION]: Enabling logins for {0}", RegionInfo.RegionName); |
1334 | // this is a rare case where we know we have just went through a long cycle of heap | 1344 | LoginsDisabled = false; |
1335 | // allocations, and there is no more work to be done until someone logs in | 1345 | m_sceneGridService.InformNeighborsThatRegionisUp(RequestModuleInterface<INeighbourService>(), RegionInfo); |
1336 | GC.Collect(); | ||
1337 | |||
1338 | IConfig startupConfig = m_config.Configs["Startup"]; | ||
1339 | if (startupConfig == null || !startupConfig.GetBoolean("StartDisabled", false)) | ||
1340 | { | ||
1341 | m_log.DebugFormat("[REGION]: Enabling logins for {0}", RegionInfo.RegionName); | ||
1342 | LoginsDisabled = false; | ||
1343 | m_sceneGridService.InformNeighborsThatRegionisUp(RequestModuleInterface<INeighbourService>(), RegionInfo); | ||
1344 | } | ||
1345 | } | 1346 | } |
1346 | } | 1347 | } |
1347 | catch (NotImplementedException) | 1348 | } |
1348 | { | 1349 | catch (NotImplementedException) |
1349 | throw; | 1350 | { |
1350 | } | 1351 | throw; |
1351 | catch (AccessViolationException e) | 1352 | } |
1352 | { | 1353 | catch (AccessViolationException e) |
1353 | m_log.Error("[REGION]: Failed with exception " + e.ToString() + " On Region: " + RegionInfo.RegionName); | 1354 | { |
1354 | } | 1355 | m_log.Error("[REGION]: Failed with exception " + e.ToString() + " On Region: " + RegionInfo.RegionName); |
1355 | //catch (NullReferenceException e) | 1356 | } |
1356 | //{ | 1357 | //catch (NullReferenceException e) |
1357 | // m_log.Error("[REGION]: Failed with exception " + e.ToString() + " On Region: " + RegionInfo.RegionName); | 1358 | //{ |
1358 | //} | 1359 | // m_log.Error("[REGION]: Failed with exception " + e.ToString() + " On Region: " + RegionInfo.RegionName); |
1359 | catch (InvalidOperationException e) | 1360 | //} |
1360 | { | 1361 | catch (InvalidOperationException e) |
1361 | m_log.Error("[REGION]: Failed with exception " + e.ToString() + " On Region: " + RegionInfo.RegionName); | 1362 | { |
1362 | } | 1363 | m_log.Error("[REGION]: Failed with exception " + e.ToString() + " On Region: " + RegionInfo.RegionName); |
1363 | catch (Exception e) | 1364 | } |
1364 | { | 1365 | catch (Exception e) |
1365 | m_log.Error("[REGION]: Failed with exception " + e.ToString() + " On Region: " + RegionInfo.RegionName); | 1366 | { |
1366 | } | 1367 | m_log.Error("[REGION]: Failed with exception " + e.ToString() + " On Region: " + RegionInfo.RegionName); |
1367 | finally | 1368 | } |
1368 | { | 1369 | finally |
1369 | m_lastupdate = DateTime.UtcNow; | 1370 | { |
1370 | } | 1371 | m_lastupdate = DateTime.UtcNow; |
1371 | 1372 | } | |
1372 | maintc = Util.EnvironmentTickCountSubtract(maintc); | ||
1373 | maintc = (int)(m_timespan * 1000) - maintc; | ||
1374 | 1373 | ||
1375 | if (maintc > 0) | 1374 | maintc = Util.EnvironmentTickCountSubtract(maintc); |
1376 | Thread.Sleep(maintc); | 1375 | maintc = (int)(m_timespan * 1000) - maintc; |
1377 | 1376 | ||
1378 | // Tell the watchdog that this thread is still alive | 1377 | if (maintc > 0) |
1379 | Watchdog.UpdateThread(); | 1378 | Thread.Sleep(maintc); |
1380 | } | ||
1381 | } | ||
1382 | 1379 | ||
1383 | 1380 | // Tell the watchdog that this thread is still alive | |
1381 | Watchdog.UpdateThread(); | ||
1382 | } | ||
1384 | 1383 | ||
1385 | public void AddGroupTarget(SceneObjectGroup grp) | 1384 | public void AddGroupTarget(SceneObjectGroup grp) |
1386 | { | 1385 | { |