mirror of
https://gitlab.com/veloren/veloren.git
synced 2024-08-30 18:12:32 +00:00
[WIP] improving thread pool usage and bot client.
This commit is contained in:
parent
3b424e9049
commit
60060a9913
9
Cargo.lock
generated
9
Cargo.lock
generated
@ -1639,6 +1639,12 @@ version = "1.2.0"
|
|||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
checksum = "9ea835d29036a4087793836fa931b08837ad5e957da9e23886b29586fb9b6650"
|
checksum = "9ea835d29036a4087793836fa931b08837ad5e957da9e23886b29586fb9b6650"
|
||||||
|
|
||||||
|
[[package]]
|
||||||
|
name = "drop_guard"
|
||||||
|
version = "0.3.0"
|
||||||
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
|
checksum = "2c4a817d8b683f6e649aed359aab0c47a875377516bb5791d0f7e46d9066d209"
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "egui"
|
name = "egui"
|
||||||
version = "0.12.0"
|
version = "0.12.0"
|
||||||
@ -6561,6 +6567,7 @@ dependencies = [
|
|||||||
"hashbrown 0.12.0",
|
"hashbrown 0.12.0",
|
||||||
"image",
|
"image",
|
||||||
"num 0.4.0",
|
"num 0.4.0",
|
||||||
|
"num_cpus",
|
||||||
"quinn",
|
"quinn",
|
||||||
"rayon",
|
"rayon",
|
||||||
"ron 0.7.0",
|
"ron 0.7.0",
|
||||||
@ -6718,6 +6725,7 @@ dependencies = [
|
|||||||
"specs",
|
"specs",
|
||||||
"tar",
|
"tar",
|
||||||
"thread-priority",
|
"thread-priority",
|
||||||
|
"tokio",
|
||||||
"toml",
|
"toml",
|
||||||
"tracing",
|
"tracing",
|
||||||
"vek 0.15.8",
|
"vek 0.15.8",
|
||||||
@ -6836,6 +6844,7 @@ dependencies = [
|
|||||||
"chrono",
|
"chrono",
|
||||||
"chrono-tz",
|
"chrono-tz",
|
||||||
"crossbeam-channel",
|
"crossbeam-channel",
|
||||||
|
"drop_guard",
|
||||||
"enumset",
|
"enumset",
|
||||||
"futures-util",
|
"futures-util",
|
||||||
"hashbrown 0.12.0",
|
"hashbrown 0.12.0",
|
||||||
|
@ -24,6 +24,7 @@ network = { package = "veloren-network", path = "../network", features = ["compr
|
|||||||
bincode = "1.3.2"
|
bincode = "1.3.2"
|
||||||
byteorder = "1.3.2"
|
byteorder = "1.3.2"
|
||||||
crossbeam-channel = "0.5"
|
crossbeam-channel = "0.5"
|
||||||
|
num_cpus = "1.0"
|
||||||
tokio = { version = "1.14", default-features = false, features = ["rt-multi-thread"] }
|
tokio = { version = "1.14", default-features = false, features = ["rt-multi-thread"] }
|
||||||
quinn = "0.8"
|
quinn = "0.8"
|
||||||
image = { version = "0.24", default-features = false, features = ["png"] }
|
image = { version = "0.24", default-features = false, features = ["png"] }
|
||||||
|
@ -50,11 +50,15 @@ fn main() {
|
|||||||
};
|
};
|
||||||
|
|
||||||
// Create a client.
|
// Create a client.
|
||||||
let mut client = runtime
|
let pools = common_state::State::pools(
|
||||||
|
common::resources::GameMode::Client,
|
||||||
|
tokio::runtime::Builder::new_multi_thread(),
|
||||||
|
);
|
||||||
|
let mut client = pools.runtime
|
||||||
.block_on(Client::new(
|
.block_on(Client::new(
|
||||||
addr,
|
addr,
|
||||||
runtime2,
|
|
||||||
&mut None,
|
&mut None,
|
||||||
|
pools.clone(),
|
||||||
&username,
|
&username,
|
||||||
&password,
|
&password,
|
||||||
|provider| provider == "https://auth.veloren.net",
|
|provider| provider == "https://auth.veloren.net",
|
||||||
|
@ -46,29 +46,28 @@ pub fn main() {
|
|||||||
|
|
||||||
pub struct BotClient {
|
pub struct BotClient {
|
||||||
settings: Settings,
|
settings: Settings,
|
||||||
runtime: Arc<Runtime>,
|
pools: common_state::Pools,
|
||||||
server_info: ServerInfo,
|
server_info: ServerInfo,
|
||||||
bot_clients: HashMap<String, Client>,
|
bot_clients: HashMap<String, Client>,
|
||||||
clock: Clock,
|
clock: Clock,
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn make_client(
|
pub fn make_client(
|
||||||
runtime: &Arc<Runtime>,
|
pools: &common_state::Pools,
|
||||||
server: &str,
|
server: &str,
|
||||||
server_info: &mut Option<ServerInfo>,
|
server_info: &mut Option<ServerInfo>,
|
||||||
username: &str,
|
username: &str,
|
||||||
password: &str,
|
password: &str,
|
||||||
) -> Option<Client> {
|
) -> Option<Client> {
|
||||||
let runtime_clone = Arc::clone(runtime);
|
|
||||||
let addr = ConnectionArgs::Tcp {
|
let addr = ConnectionArgs::Tcp {
|
||||||
prefer_ipv6: false,
|
prefer_ipv6: false,
|
||||||
hostname: server.to_owned(),
|
hostname: server.to_owned(),
|
||||||
};
|
};
|
||||||
runtime
|
pools.runtime
|
||||||
.block_on(Client::new(
|
.block_on(Client::new(
|
||||||
addr,
|
addr,
|
||||||
runtime_clone,
|
|
||||||
server_info,
|
server_info,
|
||||||
|
pools.clone(),
|
||||||
username,
|
username,
|
||||||
password,
|
password,
|
||||||
|_| true,
|
|_| true,
|
||||||
@ -78,15 +77,18 @@ pub fn make_client(
|
|||||||
|
|
||||||
impl BotClient {
|
impl BotClient {
|
||||||
pub fn new(settings: Settings) -> BotClient {
|
pub fn new(settings: Settings) -> BotClient {
|
||||||
let runtime = Arc::new(Runtime::new().unwrap());
|
let pools = common_state::State::pools(
|
||||||
|
common::resources::GameMode::Client,
|
||||||
|
tokio::runtime::Builder::new_multi_thread(),
|
||||||
|
);
|
||||||
let mut server_info = None;
|
let mut server_info = None;
|
||||||
// Don't care if we connect, just trying to grab the server info.
|
// Don't care if we connect, just trying to grab the server info.
|
||||||
let _ = make_client(&runtime, &settings.server, &mut server_info, "", "");
|
let _ = make_client(&pools, &settings.server, &mut server_info, "", "");
|
||||||
let server_info = server_info.expect("Failed to connect to server.");
|
let server_info = server_info.expect("Failed to connect to server.");
|
||||||
let clock = Clock::new(Duration::from_secs_f64(1.0 / 60.0));
|
let clock = Clock::new(Duration::from_secs_f64(1.0 / 60.0));
|
||||||
BotClient {
|
BotClient {
|
||||||
settings,
|
settings,
|
||||||
runtime,
|
pools,
|
||||||
server_info,
|
server_info,
|
||||||
bot_clients: HashMap::new(),
|
bot_clients: HashMap::new(),
|
||||||
clock,
|
clock,
|
||||||
@ -142,7 +144,7 @@ impl BotClient {
|
|||||||
{
|
{
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
match self.runtime.block_on(authc.register(username, password)) {
|
match self.pools.runtime.block_on(authc.register(username, password)) {
|
||||||
Ok(()) => {
|
Ok(()) => {
|
||||||
self.settings.bot_logins.push(BotCreds {
|
self.settings.bot_logins.push(BotCreds {
|
||||||
username: username.to_string(),
|
username: username.to_string(),
|
||||||
@ -171,15 +173,13 @@ impl BotClient {
|
|||||||
.cloned()
|
.cloned()
|
||||||
.collect();
|
.collect();
|
||||||
for cred in creds.iter() {
|
for cred in creds.iter() {
|
||||||
let runtime = Arc::clone(&self.runtime);
|
|
||||||
|
|
||||||
let server = &self.settings.server;
|
let server = &self.settings.server;
|
||||||
// TODO: log the clients in in parallel instead of in series
|
// TODO: log the clients in in parallel instead of in series
|
||||||
let client = self
|
let client = self
|
||||||
.bot_clients
|
.bot_clients
|
||||||
.entry(cred.username.clone())
|
.entry(cred.username.clone())
|
||||||
.or_insert_with(|| {
|
.or_insert_with(|| {
|
||||||
make_client(&runtime, server, &mut None, &cred.username, &cred.password)
|
make_client(&self.pools, server, &mut None, &cred.username, &cred.password)
|
||||||
.expect("Failed to connect to server")
|
.expect("Failed to connect to server")
|
||||||
});
|
});
|
||||||
|
|
||||||
|
@ -9,7 +9,7 @@ use std::{
|
|||||||
time::{Duration, SystemTime},
|
time::{Duration, SystemTime},
|
||||||
};
|
};
|
||||||
use structopt::StructOpt;
|
use structopt::StructOpt;
|
||||||
use tokio::runtime::Runtime;
|
use tokio::runtime::{self, Runtime};
|
||||||
use vek::*;
|
use vek::*;
|
||||||
use common_state::State;
|
use common_state::State;
|
||||||
use veloren_client::{addr::ConnectionArgs, Client};
|
use veloren_client::{addr::ConnectionArgs, Client};
|
||||||
@ -48,9 +48,12 @@ fn main() {
|
|||||||
let to_adminify = usernames.clone();
|
let to_adminify = usernames.clone();
|
||||||
|
|
||||||
let finished_init = Arc::new(AtomicU32::new(0));
|
let finished_init = Arc::new(AtomicU32::new(0));
|
||||||
let runtime = Arc::new(Runtime::new().unwrap());
|
let mut builder = runtime::Builder::new_multi_thread();
|
||||||
let mut pools = common_state::State::pools(common::resources::GameMode::Client);
|
builder
|
||||||
pools.0 = 0;
|
.max_blocking_threads(opt.size as usize + 1)
|
||||||
|
.thread_name("swarm");
|
||||||
|
let mut pools = common_state::State::pools(common::resources::GameMode::Client, builder);
|
||||||
|
pools.slowjob_threads = 0;
|
||||||
|
|
||||||
// TODO: calculate and log the required chunks per second to maintain the
|
// TODO: calculate and log the required chunks per second to maintain the
|
||||||
// selected scenario with full vd loaded
|
// selected scenario with full vd loaded
|
||||||
@ -59,7 +62,6 @@ fn main() {
|
|||||||
admin_username,
|
admin_username,
|
||||||
0,
|
0,
|
||||||
to_adminify,
|
to_adminify,
|
||||||
&runtime,
|
|
||||||
&pools,
|
&pools,
|
||||||
opt,
|
opt,
|
||||||
&finished_init,
|
&finished_init,
|
||||||
@ -70,7 +72,6 @@ fn main() {
|
|||||||
name,
|
name,
|
||||||
index as u32,
|
index as u32,
|
||||||
Vec::new(),
|
Vec::new(),
|
||||||
&runtime,
|
|
||||||
&pools,
|
&pools,
|
||||||
opt,
|
opt,
|
||||||
&finished_init,
|
&finished_init,
|
||||||
@ -84,16 +85,15 @@ fn run_client_new_thread(
|
|||||||
username: String,
|
username: String,
|
||||||
index: u32,
|
index: u32,
|
||||||
to_adminify: Vec<String>,
|
to_adminify: Vec<String>,
|
||||||
runtime: &Arc<Runtime>,
|
|
||||||
pools: &common_state::Pools,
|
pools: &common_state::Pools,
|
||||||
opt: Opt,
|
opt: Opt,
|
||||||
finished_init: &Arc<AtomicU32>,
|
finished_init: &Arc<AtomicU32>,
|
||||||
) {
|
) {
|
||||||
let runtime = Arc::clone(runtime);
|
let runtime = Arc::clone(&pools.runtime);
|
||||||
let pools = pools.clone();
|
let pools = pools.clone();
|
||||||
let finished_init = Arc::clone(finished_init);
|
let finished_init = Arc::clone(finished_init);
|
||||||
thread::spawn(move || {
|
thread::spawn(move || {
|
||||||
if let Err(err) = run_client(username, index, to_adminify, pools, runtime, opt, finished_init) {
|
if let Err(err) = run_client(username, index, to_adminify, pools, opt, finished_init) {
|
||||||
tracing::error!("swarm member {} exited with an error: {:?}", index, err);
|
tracing::error!("swarm member {} exited with an error: {:?}", index, err);
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
@ -104,112 +104,123 @@ fn run_client(
|
|||||||
index: u32,
|
index: u32,
|
||||||
to_adminify: Vec<String>,
|
to_adminify: Vec<String>,
|
||||||
pools: common_state::Pools,
|
pools: common_state::Pools,
|
||||||
runtime: Arc<Runtime>,
|
|
||||||
opt: Opt,
|
opt: Opt,
|
||||||
finished_init: Arc<AtomicU32>,
|
finished_init: Arc<AtomicU32>,
|
||||||
) -> Result<(), veloren_client::Error> {
|
) -> Result<(), veloren_client::Error> {
|
||||||
let mut client = loop {
|
|
||||||
// Connect to localhost
|
|
||||||
let addr = ConnectionArgs::Tcp {
|
|
||||||
prefer_ipv6: false,
|
|
||||||
hostname: "localhost".into(),
|
|
||||||
};
|
|
||||||
let runtime_clone = Arc::clone(&runtime);
|
|
||||||
// NOTE: use a no-auth server
|
|
||||||
match runtime
|
|
||||||
.block_on(Client::new(
|
|
||||||
addr,
|
|
||||||
runtime_clone,
|
|
||||||
&mut None,
|
|
||||||
pools.clone(),
|
|
||||||
&username,
|
|
||||||
"",
|
|
||||||
|_| false,
|
|
||||||
)) {
|
|
||||||
Err(e) => tracing::warn!(?e, "Client {} disconnected", index),
|
|
||||||
Ok(client) => break client,
|
|
||||||
}
|
|
||||||
};
|
|
||||||
drop(pools);
|
|
||||||
|
|
||||||
let mut clock = common::clock::Clock::new(Duration::from_secs_f32(1.0 / 30.0));
|
let mut clock = common::clock::Clock::new(Duration::from_secs_f32(1.0 / 30.0));
|
||||||
|
|
||||||
let mut tick = |client: &mut Client| -> Result<(), veloren_client::Error> {
|
let mut tick = |client: &mut Client| -> Result<(), veloren_client::Error> {
|
||||||
clock.tick();
|
clock.tick();
|
||||||
client.tick_network(clock.dt())?;
|
client.tick_network(clock.dt())?;
|
||||||
|
client.cleanup();
|
||||||
Ok(())
|
Ok(())
|
||||||
};
|
};
|
||||||
|
|
||||||
// Wait for character list to load
|
let mut run = || -> Result<_, veloren_client::Error> {
|
||||||
client.load_character_list();
|
// Connect to localhost
|
||||||
while client.character_list().loading {
|
let addr = ConnectionArgs::Tcp {
|
||||||
tick(&mut client)?;
|
prefer_ipv6: false,
|
||||||
}
|
hostname: "localhost".into(),
|
||||||
|
};
|
||||||
// Create character if none exist
|
// NOTE: use a no-auth server
|
||||||
if client.character_list().characters.is_empty() {
|
let mut client = pools.runtime.block_on(Client::new(
|
||||||
client.create_character(
|
addr,
|
||||||
username.clone(),
|
&mut None,
|
||||||
Some("common.items.weapons.sword.starter".into()),
|
pools.clone(),
|
||||||
None,
|
&username,
|
||||||
body(),
|
"",
|
||||||
);
|
|_| false,
|
||||||
|
))?;
|
||||||
|
tracing::info!("Client {} connected", index);
|
||||||
|
|
||||||
|
// Wait for character list to load
|
||||||
client.load_character_list();
|
client.load_character_list();
|
||||||
|
while client.character_list().loading {
|
||||||
while client.character_list().loading || client.character_list().characters.is_empty() {
|
|
||||||
tick(&mut client)?;
|
tick(&mut client)?;
|
||||||
}
|
}
|
||||||
}
|
tracing::info!("Client {} loaded character list", index);
|
||||||
|
|
||||||
// Select the first character
|
// Create character if none exist
|
||||||
client.request_character(
|
if client.character_list().characters.is_empty() {
|
||||||
client
|
client.create_character(
|
||||||
.character_list()
|
username.clone(),
|
||||||
.characters
|
Some("common.items.weapons.sword.starter".into()),
|
||||||
.first()
|
None,
|
||||||
.expect("Just created new character if non were listed!!!")
|
body(),
|
||||||
.character
|
);
|
||||||
.id
|
|
||||||
.expect("Why is this an option?"),
|
|
||||||
);
|
|
||||||
|
|
||||||
client.set_view_distance(opt.vd);
|
client.load_character_list();
|
||||||
|
|
||||||
// If this is the admin client then adminify the other swarm members
|
while client.character_list().loading || client.character_list().characters.is_empty() {
|
||||||
if !to_adminify.is_empty() {
|
tick(&mut client)?;
|
||||||
// Wait for other clients to connect
|
|
||||||
loop {
|
|
||||||
tick(&mut client)?;
|
|
||||||
// NOTE: it's expected that each swarm member will have a unique alias
|
|
||||||
let players = client.players().collect::<HashSet<&str>>();
|
|
||||||
if to_adminify
|
|
||||||
.iter()
|
|
||||||
.all(|name| players.contains(&name.as_str()))
|
|
||||||
{
|
|
||||||
break;
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
// Assert that we are a moderator (assumes we are an admin if so)
|
tracing::info!("Client {} found or created character", index);
|
||||||
assert!(
|
|
||||||
client.is_moderator(),
|
|
||||||
"The user needs to ensure \"{}\" is registered as an admin on the server",
|
|
||||||
username
|
|
||||||
);
|
|
||||||
// Send commands to adminify others
|
|
||||||
to_adminify.iter().for_each(|name| {
|
|
||||||
client.send_command("adminify".into(), vec![name.into(), "admin".into()])
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
// Wait for moderator
|
client.set_view_distance(opt.vd);
|
||||||
while !client.is_moderator() {
|
|
||||||
tick(&mut client)?;
|
// Select the first character
|
||||||
}
|
client.request_character(
|
||||||
|
client
|
||||||
|
.character_list()
|
||||||
|
.characters
|
||||||
|
.first()
|
||||||
|
.expect("Just created new character if non were listed!!!")
|
||||||
|
.character
|
||||||
|
.id
|
||||||
|
.expect("Why is this an option?"),
|
||||||
|
);
|
||||||
|
|
||||||
|
// If this is the admin client then adminify the other swarm members
|
||||||
|
if !to_adminify.is_empty() {
|
||||||
|
// Wait for other clients to connect
|
||||||
|
loop {
|
||||||
|
tick(&mut client)?;
|
||||||
|
// NOTE: it's expected that each swarm member will have a unique alias
|
||||||
|
let players = client.players().collect::<HashSet<&str>>();
|
||||||
|
if to_adminify
|
||||||
|
.iter()
|
||||||
|
.all(|name| players.contains(&name.as_str()))
|
||||||
|
{
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
// Assert that we are a moderator (assumes we are an admin if so)
|
||||||
|
assert!(
|
||||||
|
client.is_moderator(),
|
||||||
|
"The user needs to ensure \"{}\" is registered as an admin on the server",
|
||||||
|
username
|
||||||
|
);
|
||||||
|
// Send commands to adminify others
|
||||||
|
to_adminify.iter().for_each(|name| {
|
||||||
|
client.send_command("adminify".into(), vec![name.into(), "admin".into()])
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
// Wait for moderator
|
||||||
|
while !client.is_moderator() {
|
||||||
|
tick(&mut client)?;
|
||||||
|
}
|
||||||
|
client.clear_terrain();
|
||||||
|
client.request_player_physics(false);
|
||||||
|
|
||||||
|
Ok(client)
|
||||||
|
};
|
||||||
|
|
||||||
|
let mut client = loop {
|
||||||
|
match run() {
|
||||||
|
Err(e) => tracing::warn!(?e, "Client {} disconnected", index),
|
||||||
|
Ok(client) => {
|
||||||
|
thread::sleep(Duration::from_secs(1));
|
||||||
|
break client
|
||||||
|
},
|
||||||
|
}
|
||||||
|
};
|
||||||
|
drop(pools);
|
||||||
|
|
||||||
finished_init.fetch_add(1, Ordering::Relaxed);
|
finished_init.fetch_add(1, Ordering::Relaxed);
|
||||||
// Wait for initialization of all other swarm clients to finish
|
// Wait for initialization of all other swarm clients to finish
|
||||||
while !finished_init.load(Ordering::Relaxed) == opt.size {
|
while finished_init.load(Ordering::Relaxed) != opt.size {
|
||||||
tick(&mut client)?;
|
tick(&mut client)?;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -281,12 +281,11 @@ pub struct CharacterList {
|
|||||||
}
|
}
|
||||||
|
|
||||||
/// Higher than what's needed at VD = 65.
|
/// Higher than what's needed at VD = 65.
|
||||||
const TOTAL_PENDING_CHUNKS_LIMIT: usize = /*1024*/13800;
|
const TOTAL_PENDING_CHUNKS_LIMIT: usize = /*1024*//*13800*/12;
|
||||||
|
|
||||||
impl Client {
|
impl Client {
|
||||||
pub async fn new(
|
pub async fn new(
|
||||||
addr: ConnectionArgs,
|
addr: ConnectionArgs,
|
||||||
runtime: Arc<Runtime>,
|
|
||||||
// TODO: refactor to avoid needing to use this out parameter
|
// TODO: refactor to avoid needing to use this out parameter
|
||||||
mismatched_server_info: &mut Option<ServerInfo>,
|
mismatched_server_info: &mut Option<ServerInfo>,
|
||||||
pools: common_state::Pools,
|
pools: common_state::Pools,
|
||||||
@ -294,7 +293,8 @@ impl Client {
|
|||||||
password: &str,
|
password: &str,
|
||||||
auth_trusted: impl FnMut(&str) -> bool,
|
auth_trusted: impl FnMut(&str) -> bool,
|
||||||
) -> Result<Self, Error> {
|
) -> Result<Self, Error> {
|
||||||
let network = Network::new(Pid::new(), &runtime);
|
let network = Network::new(Pid::new(), &pools.runtime);
|
||||||
|
let runtime = Arc::clone(&pools.runtime);
|
||||||
|
|
||||||
let mut participant = match addr {
|
let mut participant = match addr {
|
||||||
ConnectionArgs::Tcp {
|
ConnectionArgs::Tcp {
|
||||||
@ -374,10 +374,324 @@ impl Client {
|
|||||||
let msg = bincode::deserialize(&msg)?;
|
let msg = bincode::deserialize(&msg)?;
|
||||||
Self::handle_server_terrain_msg(msg)
|
Self::handle_server_terrain_msg(msg)
|
||||||
};
|
};
|
||||||
terrain_tx_.send(handle_msg());
|
if terrain_tx_.send(handle_msg()).is_err() {
|
||||||
|
break;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
|
let ServerInit::GameSync {
|
||||||
|
entity_package,
|
||||||
|
time_of_day,
|
||||||
|
max_group_size,
|
||||||
|
client_timeout,
|
||||||
|
world_map,
|
||||||
|
recipe_book,
|
||||||
|
component_recipe_book,
|
||||||
|
material_stats,
|
||||||
|
ability_map,
|
||||||
|
} = loop {
|
||||||
|
tokio::select! {
|
||||||
|
// Spawn in a blocking thread (leaving the network thread free). This is mostly
|
||||||
|
// useful for bots.
|
||||||
|
res = register_stream.recv() => break res?,
|
||||||
|
_ = ping_interval.tick() => ping_stream.send(PingMsg::Ping)?,
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
// Spawn in a blocking thread (leaving the network thread free). This is mostly
|
||||||
|
// useful for bots.
|
||||||
|
let mut task = tokio::task::spawn_blocking(move || {
|
||||||
|
let map_size_lg = common::terrain::MapSizeLg::new(world_map.dimensions_lg)
|
||||||
|
.map_err(|_| {
|
||||||
|
Error::Other(format!(
|
||||||
|
"Server sent bad world map dimensions: {:?}",
|
||||||
|
world_map.dimensions_lg,
|
||||||
|
))
|
||||||
|
})?;
|
||||||
|
let sea_level = world_map.default_chunk.get_min_z() as f32;
|
||||||
|
|
||||||
|
// Initialize `State`
|
||||||
|
let mut state = State::client(pools, map_size_lg, world_map.default_chunk);
|
||||||
|
// Client-only components
|
||||||
|
state.ecs_mut().register::<comp::Last<CharacterState>>();
|
||||||
|
state.ecs_mut().write_resource::<SlowJobPool>()
|
||||||
|
.configure(&"TERRAIN_DROP", |_n| 1);
|
||||||
|
/* state.ecs_mut().write_resource::<SlowJobPool>()
|
||||||
|
.configure("TERRAIN_DESERIALIZING", |n| n / 2); */
|
||||||
|
let entity = state.ecs_mut().apply_entity_package(entity_package);
|
||||||
|
*state.ecs_mut().write_resource() = time_of_day;
|
||||||
|
*state.ecs_mut().write_resource() = PlayerEntity(Some(entity));
|
||||||
|
state.ecs_mut().insert(material_stats);
|
||||||
|
state.ecs_mut().insert(ability_map);
|
||||||
|
|
||||||
|
let map_size = map_size_lg.chunks();
|
||||||
|
let max_height = world_map.max_height;
|
||||||
|
let rgba = world_map.rgba;
|
||||||
|
let alt = world_map.alt;
|
||||||
|
if rgba.size() != map_size.map(|e| e as i32) {
|
||||||
|
return Err(Error::Other("Server sent a bad world map image".into()));
|
||||||
|
}
|
||||||
|
if alt.size() != map_size.map(|e| e as i32) {
|
||||||
|
return Err(Error::Other("Server sent a bad altitude map.".into()));
|
||||||
|
}
|
||||||
|
let [west, east] = world_map.horizons;
|
||||||
|
let scale_angle =
|
||||||
|
|a: u8| (a as f32 / 255.0 * <f32 as FloatConst>::FRAC_PI_2()).tan();
|
||||||
|
let scale_height = |h: u8| h as f32 / 255.0 * max_height;
|
||||||
|
let scale_height_big = |h: u32| (h >> 3) as f32 / 8191.0 * max_height;
|
||||||
|
|
||||||
|
debug!("Preparing image...");
|
||||||
|
let unzip_horizons = |(angles, heights): &(Vec<_>, Vec<_>)| {
|
||||||
|
(
|
||||||
|
angles.iter().copied().map(scale_angle).collect::<Vec<_>>(),
|
||||||
|
heights
|
||||||
|
.iter()
|
||||||
|
.copied()
|
||||||
|
.map(scale_height)
|
||||||
|
.collect::<Vec<_>>(),
|
||||||
|
)
|
||||||
|
};
|
||||||
|
let horizons = [unzip_horizons(&west), unzip_horizons(&east)];
|
||||||
|
|
||||||
|
// Redraw map (with shadows this time).
|
||||||
|
let mut world_map_rgba = vec![0u32; rgba.size().product() as usize];
|
||||||
|
let mut world_map_topo = vec![0u32; rgba.size().product() as usize];
|
||||||
|
let mut map_config = common::terrain::map::MapConfig::orthographic(
|
||||||
|
map_size_lg,
|
||||||
|
core::ops::RangeInclusive::new(0.0, max_height),
|
||||||
|
);
|
||||||
|
map_config.horizons = Some(&horizons);
|
||||||
|
let rescale_height = |h: f32| h / max_height;
|
||||||
|
let bounds_check = |pos: Vec2<i32>| {
|
||||||
|
pos.reduce_partial_min() >= 0
|
||||||
|
&& pos.x < map_size.x as i32
|
||||||
|
&& pos.y < map_size.y as i32
|
||||||
|
};
|
||||||
|
fn sample_pos(
|
||||||
|
map_config: &MapConfig,
|
||||||
|
pos: Vec2<i32>,
|
||||||
|
alt: &Grid<u32>,
|
||||||
|
rgba: &Grid<u32>,
|
||||||
|
map_size: &Vec2<u16>,
|
||||||
|
map_size_lg: &common::terrain::MapSizeLg,
|
||||||
|
max_height: f32,
|
||||||
|
) -> common::terrain::map::MapSample {
|
||||||
|
let rescale_height = |h: f32| h / max_height;
|
||||||
|
let scale_height_big = |h: u32| (h >> 3) as f32 / 8191.0 * max_height;
|
||||||
|
let bounds_check = |pos: Vec2<i32>| {
|
||||||
|
pos.reduce_partial_min() >= 0
|
||||||
|
&& pos.x < map_size.x as i32
|
||||||
|
&& pos.y < map_size.y as i32
|
||||||
|
};
|
||||||
|
let MapConfig {
|
||||||
|
gain,
|
||||||
|
is_contours,
|
||||||
|
is_height_map,
|
||||||
|
is_stylized_topo,
|
||||||
|
..
|
||||||
|
} = *map_config;
|
||||||
|
let mut is_contour_line = false;
|
||||||
|
let mut is_border = false;
|
||||||
|
let (rgb, alt, downhill_wpos) = if bounds_check(pos) {
|
||||||
|
let posi = pos.y as usize * map_size.x as usize + pos.x as usize;
|
||||||
|
let [r, g, b, _a] = rgba[pos].to_le_bytes();
|
||||||
|
let is_water = r == 0 && b > 102 && g < 77;
|
||||||
|
let alti = alt[pos];
|
||||||
|
// Compute contours (chunks are assigned in the river code below)
|
||||||
|
let altj = rescale_height(scale_height_big(alti));
|
||||||
|
let contour_interval = 150.0;
|
||||||
|
let chunk_contour = (altj * gain / contour_interval) as u32;
|
||||||
|
|
||||||
|
// Compute downhill.
|
||||||
|
let downhill = {
|
||||||
|
let mut best = -1;
|
||||||
|
let mut besth = alti;
|
||||||
|
for nposi in neighbors(*map_size_lg, posi) {
|
||||||
|
let nbh = alt.raw()[nposi];
|
||||||
|
let nalt = rescale_height(scale_height_big(nbh));
|
||||||
|
let nchunk_contour = (nalt * gain / contour_interval) as u32;
|
||||||
|
if !is_contour_line && chunk_contour > nchunk_contour {
|
||||||
|
is_contour_line = true;
|
||||||
|
}
|
||||||
|
let [nr, ng, nb, _na] = rgba.raw()[nposi].to_le_bytes();
|
||||||
|
let n_is_water = nr == 0 && nb > 102 && ng < 77;
|
||||||
|
|
||||||
|
if !is_border && is_water && !n_is_water {
|
||||||
|
is_border = true;
|
||||||
|
}
|
||||||
|
|
||||||
|
if nbh < besth {
|
||||||
|
besth = nbh;
|
||||||
|
best = nposi as isize;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
best
|
||||||
|
};
|
||||||
|
let downhill_wpos = if downhill < 0 {
|
||||||
|
None
|
||||||
|
} else {
|
||||||
|
Some(
|
||||||
|
Vec2::new(
|
||||||
|
(downhill as usize % map_size.x as usize) as i32,
|
||||||
|
(downhill as usize / map_size.x as usize) as i32,
|
||||||
|
) * TerrainChunkSize::RECT_SIZE.map(|e| e as i32),
|
||||||
|
)
|
||||||
|
};
|
||||||
|
(Rgb::new(r, g, b), alti, downhill_wpos)
|
||||||
|
} else {
|
||||||
|
(Rgb::zero(), 0, None)
|
||||||
|
};
|
||||||
|
let alt = f64::from(rescale_height(scale_height_big(alt)));
|
||||||
|
let wpos = pos * TerrainChunkSize::RECT_SIZE.map(|e| e as i32);
|
||||||
|
let downhill_wpos = downhill_wpos
|
||||||
|
.unwrap_or(wpos + TerrainChunkSize::RECT_SIZE.map(|e| e as i32));
|
||||||
|
let is_path = rgb.r == 0x37 && rgb.g == 0x29 && rgb.b == 0x23;
|
||||||
|
let rgb = rgb.map(|e: u8| e as f64 / 255.0);
|
||||||
|
let is_water = rgb.r == 0.0 && rgb.b > 0.4 && rgb.g < 0.3;
|
||||||
|
|
||||||
|
let rgb = if is_height_map {
|
||||||
|
if is_path {
|
||||||
|
// Path color is Rgb::new(0x37, 0x29, 0x23)
|
||||||
|
Rgb::new(0.9, 0.9, 0.63)
|
||||||
|
} else if is_water {
|
||||||
|
Rgb::new(0.23, 0.47, 0.53)
|
||||||
|
} else if is_contours && is_contour_line {
|
||||||
|
// Color contour lines
|
||||||
|
Rgb::new(0.15, 0.15, 0.15)
|
||||||
|
} else {
|
||||||
|
// Color hill shading
|
||||||
|
let lightness = (alt + 0.2).min(1.0) as f64;
|
||||||
|
Rgb::new(lightness, 0.9 * lightness, 0.5 * lightness)
|
||||||
|
}
|
||||||
|
} else if is_stylized_topo {
|
||||||
|
if is_path {
|
||||||
|
Rgb::new(0.9, 0.9, 0.63)
|
||||||
|
} else if is_water {
|
||||||
|
if is_border {
|
||||||
|
Rgb::new(0.10, 0.34, 0.50)
|
||||||
|
} else {
|
||||||
|
Rgb::new(0.23, 0.47, 0.63)
|
||||||
|
}
|
||||||
|
} else if is_contour_line {
|
||||||
|
Rgb::new(0.25, 0.25, 0.25)
|
||||||
|
} else {
|
||||||
|
// Stylized colors
|
||||||
|
Rgb::new(
|
||||||
|
(rgb.r + 0.25).min(1.0),
|
||||||
|
(rgb.g + 0.23).min(1.0),
|
||||||
|
(rgb.b + 0.10).min(1.0),
|
||||||
|
)
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
Rgb::new(rgb.r, rgb.g, rgb.b)
|
||||||
|
}
|
||||||
|
.map(|e| (e * 255.0) as u8);
|
||||||
|
common::terrain::map::MapSample {
|
||||||
|
rgb,
|
||||||
|
alt,
|
||||||
|
downhill_wpos,
|
||||||
|
connections: None,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
// Generate standard shaded map
|
||||||
|
map_config.is_shaded = true;
|
||||||
|
map_config.generate(
|
||||||
|
|pos| {
|
||||||
|
sample_pos(
|
||||||
|
&map_config,
|
||||||
|
pos,
|
||||||
|
&alt,
|
||||||
|
&rgba,
|
||||||
|
&map_size,
|
||||||
|
&map_size_lg,
|
||||||
|
max_height,
|
||||||
|
)
|
||||||
|
},
|
||||||
|
|wpos| {
|
||||||
|
let pos = wpos.map2(TerrainChunkSize::RECT_SIZE, |e, f| e / f as i32);
|
||||||
|
rescale_height(if bounds_check(pos) {
|
||||||
|
scale_height_big(alt[pos])
|
||||||
|
} else {
|
||||||
|
0.0
|
||||||
|
})
|
||||||
|
},
|
||||||
|
|pos, (r, g, b, a)| {
|
||||||
|
world_map_rgba[pos.y * map_size.x as usize + pos.x] =
|
||||||
|
u32::from_le_bytes([r, g, b, a]);
|
||||||
|
},
|
||||||
|
);
|
||||||
|
// Generate map with topographical lines and stylized colors
|
||||||
|
map_config.is_contours = true;
|
||||||
|
map_config.is_stylized_topo = true;
|
||||||
|
map_config.generate(
|
||||||
|
|pos| {
|
||||||
|
sample_pos(
|
||||||
|
&map_config,
|
||||||
|
pos,
|
||||||
|
&alt,
|
||||||
|
&rgba,
|
||||||
|
&map_size,
|
||||||
|
&map_size_lg,
|
||||||
|
max_height,
|
||||||
|
)
|
||||||
|
},
|
||||||
|
|wpos| {
|
||||||
|
let pos = wpos.map2(TerrainChunkSize::RECT_SIZE, |e, f| e / f as i32);
|
||||||
|
rescale_height(if bounds_check(pos) {
|
||||||
|
scale_height_big(alt[pos])
|
||||||
|
} else {
|
||||||
|
0.0
|
||||||
|
})
|
||||||
|
},
|
||||||
|
|pos, (r, g, b, a)| {
|
||||||
|
world_map_topo[pos.y * map_size.x as usize + pos.x] =
|
||||||
|
u32::from_le_bytes([r, g, b, a]);
|
||||||
|
},
|
||||||
|
);
|
||||||
|
let make_raw = |rgb| -> Result<_, Error> {
|
||||||
|
let mut raw = vec![0u8; 4 * world_map_rgba.len()];
|
||||||
|
LittleEndian::write_u32_into(rgb, &mut raw);
|
||||||
|
Ok(Arc::new(
|
||||||
|
DynamicImage::ImageRgba8({
|
||||||
|
// Should not fail if the dimensions are correct.
|
||||||
|
let map =
|
||||||
|
image::ImageBuffer::from_raw(u32::from(map_size.x), u32::from(map_size.y), raw);
|
||||||
|
map.ok_or_else(|| Error::Other("Server sent a bad world map image".into()))?
|
||||||
|
})
|
||||||
|
// Flip the image, since Voxygen uses an orientation where rotation from
|
||||||
|
// positive x axis to positive y axis is counterclockwise around the z axis.
|
||||||
|
.flipv(),
|
||||||
|
))
|
||||||
|
};
|
||||||
|
let lod_base = rgba;
|
||||||
|
let lod_alt = alt;
|
||||||
|
let world_map_rgb_img = make_raw(&world_map_rgba)?;
|
||||||
|
let world_map_topo_img = make_raw(&world_map_topo)?;
|
||||||
|
let world_map_layers = vec![world_map_rgb_img, world_map_topo_img];
|
||||||
|
let horizons = (west.0, west.1, east.0, east.1)
|
||||||
|
.into_par_iter()
|
||||||
|
.map(|(wa, wh, ea, eh)| u32::from_le_bytes([wa, wh, ea, eh]))
|
||||||
|
.collect::<Vec<_>>();
|
||||||
|
let lod_horizon = horizons;
|
||||||
|
let map_bounds = Vec2::new(sea_level, max_height);
|
||||||
|
debug!("Done preparing image...");
|
||||||
|
|
||||||
|
Ok((
|
||||||
|
state,
|
||||||
|
lod_base,
|
||||||
|
lod_alt,
|
||||||
|
Grid::from_raw(map_size.map(|e| e as i32), lod_horizon),
|
||||||
|
(world_map_layers, map_size, map_bounds),
|
||||||
|
world_map.sites,
|
||||||
|
world_map.pois,
|
||||||
|
recipe_book,
|
||||||
|
component_recipe_book,
|
||||||
|
max_group_size,
|
||||||
|
client_timeout,
|
||||||
|
))
|
||||||
|
});
|
||||||
|
|
||||||
let (
|
let (
|
||||||
state,
|
state,
|
||||||
lod_base,
|
lod_base,
|
||||||
@ -392,312 +706,7 @@ impl Client {
|
|||||||
client_timeout,
|
client_timeout,
|
||||||
) = loop {
|
) = loop {
|
||||||
tokio::select! {
|
tokio::select! {
|
||||||
// Spawn in a blocking thread (leaving the network thread free). This is mostly
|
res = &mut task => break res.expect("Client thread should not panic")?,
|
||||||
// useful for bots.
|
|
||||||
res = register_stream.recv() => {
|
|
||||||
let ServerInit::GameSync {
|
|
||||||
entity_package,
|
|
||||||
time_of_day,
|
|
||||||
max_group_size,
|
|
||||||
client_timeout,
|
|
||||||
world_map,
|
|
||||||
recipe_book,
|
|
||||||
component_recipe_book,
|
|
||||||
material_stats,
|
|
||||||
ability_map,
|
|
||||||
} = res?;
|
|
||||||
|
|
||||||
break tokio::task::spawn_blocking(move || {
|
|
||||||
let map_size_lg = common::terrain::MapSizeLg::new(world_map.dimensions_lg)
|
|
||||||
.map_err(|_| {
|
|
||||||
Error::Other(format!(
|
|
||||||
"Server sent bad world map dimensions: {:?}",
|
|
||||||
world_map.dimensions_lg,
|
|
||||||
))
|
|
||||||
})?;
|
|
||||||
let sea_level = world_map.default_chunk.get_min_z() as f32;
|
|
||||||
|
|
||||||
// Initialize `State`
|
|
||||||
let mut state = State::client(pools, map_size_lg, world_map.default_chunk);
|
|
||||||
// Client-only components
|
|
||||||
state.ecs_mut().register::<comp::Last<CharacterState>>();
|
|
||||||
state.ecs_mut().write_resource::<SlowJobPool>()
|
|
||||||
.configure(&"TERRAIN_DROP", |_n| 1);
|
|
||||||
/* state.ecs_mut().write_resource::<SlowJobPool>()
|
|
||||||
.configure("TERRAIN_DESERIALIZING", |n| n / 2); */
|
|
||||||
let entity = state.ecs_mut().apply_entity_package(entity_package);
|
|
||||||
*state.ecs_mut().write_resource() = time_of_day;
|
|
||||||
*state.ecs_mut().write_resource() = PlayerEntity(Some(entity));
|
|
||||||
state.ecs_mut().insert(material_stats);
|
|
||||||
state.ecs_mut().insert(ability_map);
|
|
||||||
|
|
||||||
let map_size = map_size_lg.chunks();
|
|
||||||
let max_height = world_map.max_height;
|
|
||||||
let rgba = world_map.rgba;
|
|
||||||
let alt = world_map.alt;
|
|
||||||
if rgba.size() != map_size.map(|e| e as i32) {
|
|
||||||
return Err(Error::Other("Server sent a bad world map image".into()));
|
|
||||||
}
|
|
||||||
if alt.size() != map_size.map(|e| e as i32) {
|
|
||||||
return Err(Error::Other("Server sent a bad altitude map.".into()));
|
|
||||||
}
|
|
||||||
let [west, east] = world_map.horizons;
|
|
||||||
let scale_angle =
|
|
||||||
|a: u8| (a as f32 / 255.0 * <f32 as FloatConst>::FRAC_PI_2()).tan();
|
|
||||||
let scale_height = |h: u8| h as f32 / 255.0 * max_height;
|
|
||||||
let scale_height_big = |h: u32| (h >> 3) as f32 / 8191.0 * max_height;
|
|
||||||
|
|
||||||
debug!("Preparing image...");
|
|
||||||
let unzip_horizons = |(angles, heights): &(Vec<_>, Vec<_>)| {
|
|
||||||
(
|
|
||||||
angles.iter().copied().map(scale_angle).collect::<Vec<_>>(),
|
|
||||||
heights
|
|
||||||
.iter()
|
|
||||||
.copied()
|
|
||||||
.map(scale_height)
|
|
||||||
.collect::<Vec<_>>(),
|
|
||||||
)
|
|
||||||
};
|
|
||||||
let horizons = [unzip_horizons(&west), unzip_horizons(&east)];
|
|
||||||
|
|
||||||
// Redraw map (with shadows this time).
|
|
||||||
let mut world_map_rgba = vec![0u32; rgba.size().product() as usize];
|
|
||||||
let mut world_map_topo = vec![0u32; rgba.size().product() as usize];
|
|
||||||
let mut map_config = common::terrain::map::MapConfig::orthographic(
|
|
||||||
map_size_lg,
|
|
||||||
core::ops::RangeInclusive::new(0.0, max_height),
|
|
||||||
);
|
|
||||||
map_config.horizons = Some(&horizons);
|
|
||||||
let rescale_height = |h: f32| h / max_height;
|
|
||||||
let bounds_check = |pos: Vec2<i32>| {
|
|
||||||
pos.reduce_partial_min() >= 0
|
|
||||||
&& pos.x < map_size.x as i32
|
|
||||||
&& pos.y < map_size.y as i32
|
|
||||||
};
|
|
||||||
fn sample_pos(
|
|
||||||
map_config: &MapConfig,
|
|
||||||
pos: Vec2<i32>,
|
|
||||||
alt: &Grid<u32>,
|
|
||||||
rgba: &Grid<u32>,
|
|
||||||
map_size: &Vec2<u16>,
|
|
||||||
map_size_lg: &common::terrain::MapSizeLg,
|
|
||||||
max_height: f32,
|
|
||||||
) -> common::terrain::map::MapSample {
|
|
||||||
let rescale_height = |h: f32| h / max_height;
|
|
||||||
let scale_height_big = |h: u32| (h >> 3) as f32 / 8191.0 * max_height;
|
|
||||||
let bounds_check = |pos: Vec2<i32>| {
|
|
||||||
pos.reduce_partial_min() >= 0
|
|
||||||
&& pos.x < map_size.x as i32
|
|
||||||
&& pos.y < map_size.y as i32
|
|
||||||
};
|
|
||||||
let MapConfig {
|
|
||||||
gain,
|
|
||||||
is_contours,
|
|
||||||
is_height_map,
|
|
||||||
is_stylized_topo,
|
|
||||||
..
|
|
||||||
} = *map_config;
|
|
||||||
let mut is_contour_line = false;
|
|
||||||
let mut is_border = false;
|
|
||||||
let (rgb, alt, downhill_wpos) = if bounds_check(pos) {
|
|
||||||
let posi = pos.y as usize * map_size.x as usize + pos.x as usize;
|
|
||||||
let [r, g, b, _a] = rgba[pos].to_le_bytes();
|
|
||||||
let is_water = r == 0 && b > 102 && g < 77;
|
|
||||||
let alti = alt[pos];
|
|
||||||
// Compute contours (chunks are assigned in the river code below)
|
|
||||||
let altj = rescale_height(scale_height_big(alti));
|
|
||||||
let contour_interval = 150.0;
|
|
||||||
let chunk_contour = (altj * gain / contour_interval) as u32;
|
|
||||||
|
|
||||||
// Compute downhill.
|
|
||||||
let downhill = {
|
|
||||||
let mut best = -1;
|
|
||||||
let mut besth = alti;
|
|
||||||
for nposi in neighbors(*map_size_lg, posi) {
|
|
||||||
let nbh = alt.raw()[nposi];
|
|
||||||
let nalt = rescale_height(scale_height_big(nbh));
|
|
||||||
let nchunk_contour = (nalt * gain / contour_interval) as u32;
|
|
||||||
if !is_contour_line && chunk_contour > nchunk_contour {
|
|
||||||
is_contour_line = true;
|
|
||||||
}
|
|
||||||
let [nr, ng, nb, _na] = rgba.raw()[nposi].to_le_bytes();
|
|
||||||
let n_is_water = nr == 0 && nb > 102 && ng < 77;
|
|
||||||
|
|
||||||
if !is_border && is_water && !n_is_water {
|
|
||||||
is_border = true;
|
|
||||||
}
|
|
||||||
|
|
||||||
if nbh < besth {
|
|
||||||
besth = nbh;
|
|
||||||
best = nposi as isize;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
best
|
|
||||||
};
|
|
||||||
let downhill_wpos = if downhill < 0 {
|
|
||||||
None
|
|
||||||
} else {
|
|
||||||
Some(
|
|
||||||
Vec2::new(
|
|
||||||
(downhill as usize % map_size.x as usize) as i32,
|
|
||||||
(downhill as usize / map_size.x as usize) as i32,
|
|
||||||
) * TerrainChunkSize::RECT_SIZE.map(|e| e as i32),
|
|
||||||
)
|
|
||||||
};
|
|
||||||
(Rgb::new(r, g, b), alti, downhill_wpos)
|
|
||||||
} else {
|
|
||||||
(Rgb::zero(), 0, None)
|
|
||||||
};
|
|
||||||
let alt = f64::from(rescale_height(scale_height_big(alt)));
|
|
||||||
let wpos = pos * TerrainChunkSize::RECT_SIZE.map(|e| e as i32);
|
|
||||||
let downhill_wpos = downhill_wpos
|
|
||||||
.unwrap_or(wpos + TerrainChunkSize::RECT_SIZE.map(|e| e as i32));
|
|
||||||
let is_path = rgb.r == 0x37 && rgb.g == 0x29 && rgb.b == 0x23;
|
|
||||||
let rgb = rgb.map(|e: u8| e as f64 / 255.0);
|
|
||||||
let is_water = rgb.r == 0.0 && rgb.b > 0.4 && rgb.g < 0.3;
|
|
||||||
|
|
||||||
let rgb = if is_height_map {
|
|
||||||
if is_path {
|
|
||||||
// Path color is Rgb::new(0x37, 0x29, 0x23)
|
|
||||||
Rgb::new(0.9, 0.9, 0.63)
|
|
||||||
} else if is_water {
|
|
||||||
Rgb::new(0.23, 0.47, 0.53)
|
|
||||||
} else if is_contours && is_contour_line {
|
|
||||||
// Color contour lines
|
|
||||||
Rgb::new(0.15, 0.15, 0.15)
|
|
||||||
} else {
|
|
||||||
// Color hill shading
|
|
||||||
let lightness = (alt + 0.2).min(1.0) as f64;
|
|
||||||
Rgb::new(lightness, 0.9 * lightness, 0.5 * lightness)
|
|
||||||
}
|
|
||||||
} else if is_stylized_topo {
|
|
||||||
if is_path {
|
|
||||||
Rgb::new(0.9, 0.9, 0.63)
|
|
||||||
} else if is_water {
|
|
||||||
if is_border {
|
|
||||||
Rgb::new(0.10, 0.34, 0.50)
|
|
||||||
} else {
|
|
||||||
Rgb::new(0.23, 0.47, 0.63)
|
|
||||||
}
|
|
||||||
} else if is_contour_line {
|
|
||||||
Rgb::new(0.25, 0.25, 0.25)
|
|
||||||
} else {
|
|
||||||
// Stylized colors
|
|
||||||
Rgb::new(
|
|
||||||
(rgb.r + 0.25).min(1.0),
|
|
||||||
(rgb.g + 0.23).min(1.0),
|
|
||||||
(rgb.b + 0.10).min(1.0),
|
|
||||||
)
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
Rgb::new(rgb.r, rgb.g, rgb.b)
|
|
||||||
}
|
|
||||||
.map(|e| (e * 255.0) as u8);
|
|
||||||
common::terrain::map::MapSample {
|
|
||||||
rgb,
|
|
||||||
alt,
|
|
||||||
downhill_wpos,
|
|
||||||
connections: None,
|
|
||||||
}
|
|
||||||
}
|
|
||||||
// Generate standard shaded map
|
|
||||||
map_config.is_shaded = true;
|
|
||||||
map_config.generate(
|
|
||||||
|pos| {
|
|
||||||
sample_pos(
|
|
||||||
&map_config,
|
|
||||||
pos,
|
|
||||||
&alt,
|
|
||||||
&rgba,
|
|
||||||
&map_size,
|
|
||||||
&map_size_lg,
|
|
||||||
max_height,
|
|
||||||
)
|
|
||||||
},
|
|
||||||
|wpos| {
|
|
||||||
let pos = wpos.map2(TerrainChunkSize::RECT_SIZE, |e, f| e / f as i32);
|
|
||||||
rescale_height(if bounds_check(pos) {
|
|
||||||
scale_height_big(alt[pos])
|
|
||||||
} else {
|
|
||||||
0.0
|
|
||||||
})
|
|
||||||
},
|
|
||||||
|pos, (r, g, b, a)| {
|
|
||||||
world_map_rgba[pos.y * map_size.x as usize + pos.x] =
|
|
||||||
u32::from_le_bytes([r, g, b, a]);
|
|
||||||
},
|
|
||||||
);
|
|
||||||
// Generate map with topographical lines and stylized colors
|
|
||||||
map_config.is_contours = true;
|
|
||||||
map_config.is_stylized_topo = true;
|
|
||||||
map_config.generate(
|
|
||||||
|pos| {
|
|
||||||
sample_pos(
|
|
||||||
&map_config,
|
|
||||||
pos,
|
|
||||||
&alt,
|
|
||||||
&rgba,
|
|
||||||
&map_size,
|
|
||||||
&map_size_lg,
|
|
||||||
max_height,
|
|
||||||
)
|
|
||||||
},
|
|
||||||
|wpos| {
|
|
||||||
let pos = wpos.map2(TerrainChunkSize::RECT_SIZE, |e, f| e / f as i32);
|
|
||||||
rescale_height(if bounds_check(pos) {
|
|
||||||
scale_height_big(alt[pos])
|
|
||||||
} else {
|
|
||||||
0.0
|
|
||||||
})
|
|
||||||
},
|
|
||||||
|pos, (r, g, b, a)| {
|
|
||||||
world_map_topo[pos.y * map_size.x as usize + pos.x] =
|
|
||||||
u32::from_le_bytes([r, g, b, a]);
|
|
||||||
},
|
|
||||||
);
|
|
||||||
let make_raw = |rgb| -> Result<_, Error> {
|
|
||||||
let mut raw = vec![0u8; 4 * world_map_rgba.len()];
|
|
||||||
LittleEndian::write_u32_into(rgb, &mut raw);
|
|
||||||
Ok(Arc::new(
|
|
||||||
DynamicImage::ImageRgba8({
|
|
||||||
// Should not fail if the dimensions are correct.
|
|
||||||
let map =
|
|
||||||
image::ImageBuffer::from_raw(u32::from(map_size.x), u32::from(map_size.y), raw);
|
|
||||||
map.ok_or_else(|| Error::Other("Server sent a bad world map image".into()))?
|
|
||||||
})
|
|
||||||
// Flip the image, since Voxygen uses an orientation where rotation from
|
|
||||||
// positive x axis to positive y axis is counterclockwise around the z axis.
|
|
||||||
.flipv(),
|
|
||||||
))
|
|
||||||
};
|
|
||||||
let lod_base = rgba;
|
|
||||||
let lod_alt = alt;
|
|
||||||
let world_map_rgb_img = make_raw(&world_map_rgba)?;
|
|
||||||
let world_map_topo_img = make_raw(&world_map_topo)?;
|
|
||||||
let world_map_layers = vec![world_map_rgb_img, world_map_topo_img];
|
|
||||||
let horizons = (west.0, west.1, east.0, east.1)
|
|
||||||
.into_par_iter()
|
|
||||||
.map(|(wa, wh, ea, eh)| u32::from_le_bytes([wa, wh, ea, eh]))
|
|
||||||
.collect::<Vec<_>>();
|
|
||||||
let lod_horizon = horizons;
|
|
||||||
let map_bounds = Vec2::new(sea_level, max_height);
|
|
||||||
debug!("Done preparing image...");
|
|
||||||
|
|
||||||
Ok((
|
|
||||||
state,
|
|
||||||
lod_base,
|
|
||||||
lod_alt,
|
|
||||||
Grid::from_raw(map_size.map(|e| e as i32), lod_horizon),
|
|
||||||
(world_map_layers, map_size, map_bounds),
|
|
||||||
world_map.sites,
|
|
||||||
world_map.pois,
|
|
||||||
recipe_book,
|
|
||||||
component_recipe_book,
|
|
||||||
max_group_size,
|
|
||||||
client_timeout,
|
|
||||||
))
|
|
||||||
}).await.expect("Client thread should not panic")?;
|
|
||||||
},
|
|
||||||
_ = ping_interval.tick() => ping_stream.send(PingMsg::Ping)?,
|
_ = ping_interval.tick() => ping_stream.send(PingMsg::Ping)?,
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
@ -1900,7 +1909,7 @@ impl Client {
|
|||||||
+ TerrainChunkSize::RECT_SIZE.map(|x| x as f32) / 2.0)
|
+ TerrainChunkSize::RECT_SIZE.map(|x| x as f32) / 2.0)
|
||||||
.distance_squared(pos.0.into());
|
.distance_squared(pos.0.into());
|
||||||
|
|
||||||
let mut terrain = self.state.terrain();
|
let terrain = self.state.terrain();
|
||||||
if let Some(chunk) = terrain.get_key_arc(*key) {
|
if let Some(chunk) = terrain.get_key_arc(*key) {
|
||||||
if !skip_mode && !terrain.contains_key_real(*key) {
|
if !skip_mode && !terrain.contains_key_real(*key) {
|
||||||
let chunk = Arc::clone(chunk);
|
let chunk = Arc::clone(chunk);
|
||||||
@ -1910,10 +1919,10 @@ impl Client {
|
|||||||
} else {
|
} else {
|
||||||
drop(terrain);
|
drop(terrain);
|
||||||
if !skip_mode && !self.pending_chunks.contains_key(key) {
|
if !skip_mode && !self.pending_chunks.contains_key(key) {
|
||||||
const CURRENT_TICK_PENDING_CHUNKS_LIMIT: usize = 8 * 4;
|
const CURRENT_TICK_PENDING_CHUNKS_LIMIT: usize = /*8 * 4*/2;
|
||||||
if self.pending_chunks.len() < TOTAL_PENDING_CHUNKS_LIMIT
|
if self.pending_chunks.len() < TOTAL_PENDING_CHUNKS_LIMIT
|
||||||
&& /* current_tick_send_chunk_requests
|
&& current_tick_send_chunk_requests
|
||||||
< CURRENT_TICK_PENDING_CHUNKS_LIMIT */true
|
< CURRENT_TICK_PENDING_CHUNKS_LIMIT
|
||||||
{
|
{
|
||||||
self.send_msg_err(ClientGeneral::TerrainChunkRequest {
|
self.send_msg_err(ClientGeneral::TerrainChunkRequest {
|
||||||
key: *key,
|
key: *key,
|
||||||
@ -2477,7 +2486,22 @@ impl Client {
|
|||||||
{
|
{
|
||||||
terrain_cnt += msg.size();
|
terrain_cnt += msg.size();
|
||||||
}
|
}
|
||||||
self.terrain_tx.send(msg);
|
match self.terrain_tx.try_send(msg) {
|
||||||
|
Ok(()) => {},
|
||||||
|
Err(mpsc::TrySendError::Full(msg)) => {
|
||||||
|
// The message thread is fully backed up, which means we can blocking
|
||||||
|
// process at least the size of the channel in chunk receives. For now, to
|
||||||
|
// avoid blocking for too long, we just ask for a single chunk. Note that
|
||||||
|
// since the thread is processed serially, try_recv would actually be
|
||||||
|
// guaranteed to succeed here, but we avoid relying on that assumption.
|
||||||
|
let msg_ = self.terrain_rx.recv()
|
||||||
|
.expect("Deserialization thread shouold not panic")?;
|
||||||
|
self.handle_terrain_msg(msg_);
|
||||||
|
self.terrain_tx.send(msg).expect("Deserialization thread should not panic");
|
||||||
|
},
|
||||||
|
Err(mpsc::TrySendError::Disconnected(_)) =>
|
||||||
|
unreachable!("Deserialization thread should not panic."),
|
||||||
|
}
|
||||||
}
|
}
|
||||||
/* if !terrain_messages.is_empty() {
|
/* if !terrain_messages.is_empty() {
|
||||||
cnt += terrain_messages.len() as u64;
|
cnt += terrain_messages.len() as u64;
|
||||||
@ -2537,8 +2561,9 @@ impl Client {
|
|||||||
return Err(Error::ServerTimeout);
|
return Err(Error::ServerTimeout);
|
||||||
}
|
}
|
||||||
|
|
||||||
// ignore network events
|
|
||||||
while let Some(Ok(Some(event))) = self.participant.as_mut().map(|p| p.try_fetch_event()) {
|
while let Some(res) = self.participant.as_mut().and_then(|p| p.try_fetch_event().transpose()) {
|
||||||
|
let event = res?;
|
||||||
trace!(?event, "received network event");
|
trace!(?event, "received network event");
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -2868,6 +2893,12 @@ impl Client {
|
|||||||
// Handle new messages from the server.
|
// Handle new messages from the server.
|
||||||
self.handle_new_messages()?;
|
self.handle_new_messages()?;
|
||||||
|
|
||||||
|
// TODO: avoid emitting these in the first place
|
||||||
|
self.state
|
||||||
|
.ecs()
|
||||||
|
.fetch::<EventBus<common::event::ServerEvent>>()
|
||||||
|
.recv_all();
|
||||||
|
|
||||||
// 5) Terrain
|
// 5) Terrain
|
||||||
self.tick_terrain()?;
|
self.tick_terrain()?;
|
||||||
let empty = Arc::new(TerrainChunk::new(
|
let empty = Arc::new(TerrainChunk::new(
|
||||||
|
@ -20,6 +20,7 @@ core_affinity = "0.5"
|
|||||||
rayon = "1.5"
|
rayon = "1.5"
|
||||||
num_cpus = "1.0"
|
num_cpus = "1.0"
|
||||||
thread-priority = { version = "0.9.2" }
|
thread-priority = { version = "0.9.2" }
|
||||||
|
tokio = { version = "1.14", default-features = false, features = ["rt"] }
|
||||||
tracing = { version = "0.1", default-features = false }
|
tracing = { version = "0.1", default-features = false }
|
||||||
vek = { version = "0.15.8", features = ["serde"] }
|
vek = { version = "0.15.8", features = ["serde"] }
|
||||||
|
|
||||||
|
@ -36,7 +36,7 @@ use specs::{
|
|||||||
Component, DispatcherBuilder, Entity as EcsEntity, WorldExt,
|
Component, DispatcherBuilder, Entity as EcsEntity, WorldExt,
|
||||||
};
|
};
|
||||||
use thread_priority::{ThreadBuilder, ThreadPriority};
|
use thread_priority::{ThreadBuilder, ThreadPriority};
|
||||||
use std::sync::Arc;
|
use std::sync::{atomic::{AtomicUsize, Ordering}, Arc};
|
||||||
use vek::*;
|
use vek::*;
|
||||||
|
|
||||||
/// How much faster should an in-game day be compared to a real day?
|
/// How much faster should an in-game day be compared to a real day?
|
||||||
@ -95,36 +95,50 @@ pub struct State {
|
|||||||
thread_pool: Arc<ThreadPool>,
|
thread_pool: Arc<ThreadPool>,
|
||||||
}
|
}
|
||||||
|
|
||||||
pub type Pools = (usize, GameMode/*u64*/, Arc<ThreadPool>/*, slowjob::SlowJobPool*/);
|
#[derive(Clone,Debug)]
|
||||||
|
pub struct Pools {
|
||||||
|
pub slowjob_threads: usize,
|
||||||
|
game_mode: GameMode,
|
||||||
|
pub rayon_pool: Arc<ThreadPool>,
|
||||||
|
pub runtime: Arc<tokio::runtime::Runtime>,
|
||||||
|
/* slowjob: slowjob::SlowJobPool,*/
|
||||||
|
}
|
||||||
|
|
||||||
impl State {
|
impl State {
|
||||||
pub fn pools(game_mode: GameMode) -> Pools {
|
pub fn pools(game_mode: GameMode, mut tokio_builder: tokio::runtime::Builder) -> Pools {
|
||||||
let num_cpu = num_cpus::get()/* - 1*/;
|
let num_cpu = num_cpus::get()/* - 1*/;
|
||||||
|
|
||||||
let (thread_name_infix, rayon_offset) = match game_mode {
|
let (thread_name_infix, rayon_offset, tokio_count) = match game_mode {
|
||||||
// The server does work on both the main thread and the rayon pool, but not at the same
|
// The server does work on both the main thread and the rayon pool, but not at the same
|
||||||
// time, so there's no reason to hold back a core from rayon. It does effectively no
|
// time, so there's no reason to hold back a core from rayon. It does effectively no
|
||||||
// other non-slowjob, non-rayon work that blocks the main thread.
|
// other non-slowjob, non-rayon work that blocks the main thread, but we do want to
|
||||||
GameMode::Server => ("s", 0),
|
// leave num_cpu / 4 cores available for tokio, since otherwise TCP buffers can build
|
||||||
|
// up to unreasonable degrees.
|
||||||
|
GameMode::Server => ("s", 0, num_cpu / 4),
|
||||||
// The client does work on both the main thread and the rayon pool, but not at the same
|
// The client does work on both the main thread and the rayon pool, but not at the same
|
||||||
// time. It does run some other non-slowjob and non-rayon threads, but none of them
|
// time. It does run some other non-slowjob and non-rayon threads, but none of them
|
||||||
// block work on the main thread.
|
// block work on the main thread. It does not do enough networking for it to be worth
|
||||||
GameMode::Client => ("c", 0),
|
// dedicating anything to the tokio pool.
|
||||||
|
GameMode::Client => ("c", 0, 0),
|
||||||
// Singleplayer does work on both the main thread and the rayon pool; unlike the server
|
// Singleplayer does work on both the main thread and the rayon pool; unlike the server
|
||||||
// and client cases, the rayon work may interfere with work on one of the main threads,
|
// and client cases, the rayon work may interfere with work on one of the main threads,
|
||||||
// since the server and client don't coordinate their rayon work. Therefore, we
|
// since the server and client don't coordinate their rayon work. Therefore, we
|
||||||
// reserve a core for the main thread(s) from both slowjob and rayon tasks. Since many
|
// reserve a core for the main thread(s) from both slowjob and rayon tasks. Since many
|
||||||
// CPUs can't afford to lose an entire core during the common periods when the server
|
// CPUs can't afford to lose an entire core during the common periods when the server
|
||||||
// and client are not interfering with each other, we still spawn an extra rayon thread
|
// and client are not interfering with each other, we still spawn an extra rayon thread
|
||||||
// in this case, but leave it floating so the OS can schedule it.
|
// in this case, but leave it floating so the OS can schedule it. Of course,
|
||||||
GameMode::Singleplayer => ("sp", /*2*/1),
|
// singleplayer need not devote any resources to the tokio pool, as it's not expected
|
||||||
|
// to be in use.
|
||||||
|
GameMode::Singleplayer => ("sp", /*2*/1, 0),
|
||||||
};
|
};
|
||||||
let rayon_threads = /*match game_mode {
|
let rayon_threads = /*match game_mode {
|
||||||
GameMode::Server | GameMode::Client => num_cpu / 2,
|
GameMode::Server | GameMode::Client => num_cpu / 2,
|
||||||
GameMode::Singleplayer => num_cpu / 4,
|
GameMode::Singleplayer => num_cpu / 4,
|
||||||
}*/num_cpu.max(common::consts::MIN_RECOMMENDED_RAYON_THREADS);
|
}*/num_cpu/*.saturating_sub(tokio_count)*/.max(common::consts::MIN_RECOMMENDED_RAYON_THREADS);
|
||||||
let core_ids = /*(rayon_threads >= 16).then(|| */core_affinity::get_core_ids().unwrap_or(vec![])/*).unwrap_or(vec![])*/;
|
let core_ids = /*(rayon_threads >= 16).then(|| */core_affinity::get_core_ids().unwrap_or(vec![])/*).unwrap_or(vec![])*/;
|
||||||
|
// Don't pin rayon threads to the cores expected to be used by tokio threads.
|
||||||
|
/* core_ids.truncate(core_ids.len() - tokio_count); */
|
||||||
|
let mut core_ids_ = core_ids.clone();
|
||||||
let core_count = core_ids.len();
|
let core_count = core_ids.len();
|
||||||
let rayon_pool = Arc::new(
|
let rayon_pool = Arc::new(
|
||||||
ThreadPoolBuilder::new()
|
ThreadPoolBuilder::new()
|
||||||
@ -141,14 +155,14 @@ impl State {
|
|||||||
}
|
}
|
||||||
// pinned rayon threads run with high priority
|
// pinned rayon threads run with high priority
|
||||||
let index = thread.index();
|
let index = thread.index();
|
||||||
if index.checked_sub(rayon_offset).map_or(false, |i| i < core_count) {
|
if index.checked_sub(rayon_offset + tokio_count).map_or(false, |i| i < core_count) {
|
||||||
b = b.priority(ThreadPriority::Max);
|
b = b.priority(ThreadPriority::Max);
|
||||||
}
|
}
|
||||||
b.spawn_careless(|| thread.run())?;
|
b.spawn_careless(|| thread.run())?;
|
||||||
Ok(())
|
Ok(())
|
||||||
})
|
})
|
||||||
.start_handler(move |i| {
|
.start_handler(move |i| {
|
||||||
if let Some(&core_id) = i.checked_sub(rayon_offset).and_then(|i| core_ids.get(i)) {
|
if let Some(&core_id) = i.checked_sub(rayon_offset + tokio_count).and_then(|i| core_ids_.get(i)) {
|
||||||
core_affinity::set_for_current(core_id);
|
core_affinity::set_for_current(core_id);
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
@ -173,9 +187,49 @@ impl State {
|
|||||||
/*Arc::clone(*/slow_pool/*)*/,
|
/*Arc::clone(*/slow_pool/*)*/,
|
||||||
); */
|
); */
|
||||||
|
|
||||||
|
// Setup tokio runtime
|
||||||
|
use tokio::runtime::Builder;
|
||||||
|
|
||||||
|
// TODO: evaluate std::thread::available_concurrency as a num_cpus replacement
|
||||||
|
let cores = num_cpus::get();
|
||||||
|
// We don't need that many threads in the async pool, at least 2 but generally
|
||||||
|
// 25% of all available will do
|
||||||
|
let mut core_ids_ = core_ids.clone();
|
||||||
|
let runtime = Arc::new(
|
||||||
|
tokio_builder
|
||||||
|
.enable_all()
|
||||||
|
.worker_threads(tokio_count.max(common::consts::MIN_RECOMMENDED_TOKIO_THREADS))
|
||||||
|
.thread_name_fn(move || {
|
||||||
|
static ATOMIC_ID: AtomicUsize = AtomicUsize::new(0);
|
||||||
|
let id = ATOMIC_ID.fetch_add(1, Ordering::SeqCst);
|
||||||
|
format!("tokio-{}-{}", thread_name_infix, id)
|
||||||
|
})
|
||||||
|
.on_thread_start(move || {
|
||||||
|
if tokio_count > 0 {
|
||||||
|
static ATOMIC_ID: AtomicUsize = AtomicUsize::new(0);
|
||||||
|
let index = ATOMIC_ID.fetch_add(1, Ordering::SeqCst) % tokio_count;
|
||||||
|
if let Some(&core_id) = index.checked_add(num_cpu - tokio_count - 1).and_then(|i| core_ids_.get(i)) {
|
||||||
|
core_affinity::set_for_current(core_id);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
})
|
||||||
|
.build()
|
||||||
|
.unwrap(),
|
||||||
|
);
|
||||||
|
|
||||||
// We always reserve at least one non-slowjob core, if possible, to make sure systems get a
|
// We always reserve at least one non-slowjob core, if possible, to make sure systems get a
|
||||||
// chance to run unobstructed.
|
// chance to run unobstructed. We also leave any dedicated tokio threads their own
|
||||||
(num_cpu - 1/*slow_limit*//* as u64*/, game_mode, rayon_pool/*, slowjob*/)
|
// reserved cores, since networking is quite latency critical on a server.
|
||||||
|
//
|
||||||
|
// Note that for all n > 1, n - n / 4 - 1 > 0 (using integer arithmetic), which is fine
|
||||||
|
// since we require at least 2 hardware threads.
|
||||||
|
Pools {
|
||||||
|
slowjob_threads: num_cpu - tokio_count - 1/*slow_limit*//* as u64*/,
|
||||||
|
game_mode,
|
||||||
|
rayon_pool,
|
||||||
|
runtime,
|
||||||
|
/*slowjob,*/
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Create a new `State` in client mode.
|
/// Create a new `State` in client mode.
|
||||||
@ -195,8 +249,8 @@ impl State {
|
|||||||
GameMode::Singleplayer => "sp",
|
GameMode::Singleplayer => "sp",
|
||||||
}; */
|
}; */
|
||||||
|
|
||||||
let num_cpu = /*num_cpus::get()*/pools.0/* / 2 + pools.0 / 4*/;
|
let num_cpu = /*num_cpus::get()*/pools.slowjob_threads/* / 2 + pools.0 / 4*/;
|
||||||
let game_mode = pools.1;
|
let game_mode = pools.game_mode;
|
||||||
/* let rayon_threads = match game_mode {
|
/* let rayon_threads = match game_mode {
|
||||||
GameMode::Server | GameMode::Client => num_cpu/* / 2*/,
|
GameMode::Server | GameMode::Client => num_cpu/* / 2*/,
|
||||||
GameMode::Singleplayer => num_cpu/* / 4*// 2,
|
GameMode::Singleplayer => num_cpu/* / 4*// 2,
|
||||||
@ -248,7 +302,7 @@ impl State {
|
|||||||
|
|
||||||
Self {
|
Self {
|
||||||
ecs: Self::setup_ecs_world(ecs_role, /*num_cpu as u64*//*, &thread_pool, *//*pools.1*/slowjob/*pools.3*/, map_size_lg, default_chunk),
|
ecs: Self::setup_ecs_world(ecs_role, /*num_cpu as u64*//*, &thread_pool, *//*pools.1*/slowjob/*pools.3*/, map_size_lg, default_chunk),
|
||||||
thread_pool: pools.2,
|
thread_pool: pools.rayon_pool,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -18,9 +18,8 @@ use crate::{
|
|||||||
tui_runner::Tui,
|
tui_runner::Tui,
|
||||||
tuilog::TuiLog,
|
tuilog::TuiLog,
|
||||||
};
|
};
|
||||||
use common::{clock::Clock, consts::MIN_RECOMMENDED_TOKIO_THREADS};
|
use common::clock::Clock;
|
||||||
use common_base::span;
|
use common_base::span;
|
||||||
use core::sync::atomic::{AtomicUsize, Ordering};
|
|
||||||
use server::{persistence::DatabaseSettings, settings::Protocol, Event, Input, Server};
|
use server::{persistence::DatabaseSettings, settings::Protocol, Event, Input, Server};
|
||||||
use std::{
|
use std::{
|
||||||
io,
|
io,
|
||||||
@ -71,20 +70,9 @@ fn main() -> io::Result<()> {
|
|||||||
path
|
path
|
||||||
};
|
};
|
||||||
|
|
||||||
// We don't need that many threads in the async pool, at least 2 but generally
|
let pools = common_state::State::pools(
|
||||||
// 25% of all available will do
|
common::resources::GameMode::Server,
|
||||||
// TODO: evaluate std::thread::available_concurrency as a num_cpus replacement
|
tokio::runtime::Builder::new_multi_thread(),
|
||||||
let runtime = Arc::new(
|
|
||||||
tokio::runtime::Builder::new_multi_thread()
|
|
||||||
.enable_all()
|
|
||||||
.worker_threads((num_cpus::get() / 4).max(MIN_RECOMMENDED_TOKIO_THREADS))
|
|
||||||
.thread_name_fn(|| {
|
|
||||||
static ATOMIC_ID: AtomicUsize = AtomicUsize::new(0);
|
|
||||||
let id = ATOMIC_ID.fetch_add(1, Ordering::SeqCst);
|
|
||||||
format!("tokio-server-{}", id)
|
|
||||||
})
|
|
||||||
.build()
|
|
||||||
.unwrap(),
|
|
||||||
);
|
);
|
||||||
|
|
||||||
// Load server settings
|
// Load server settings
|
||||||
@ -109,7 +97,7 @@ fn main() -> io::Result<()> {
|
|||||||
ArgvCommand::Shared(SharedCommand::Admin { command }) => {
|
ArgvCommand::Shared(SharedCommand::Admin { command }) => {
|
||||||
let login_provider = server::login_provider::LoginProvider::new(
|
let login_provider = server::login_provider::LoginProvider::new(
|
||||||
server_settings.auth_server_address,
|
server_settings.auth_server_address,
|
||||||
runtime,
|
pools.runtime,
|
||||||
);
|
);
|
||||||
|
|
||||||
match command {
|
match command {
|
||||||
@ -168,8 +156,7 @@ fn main() -> io::Result<()> {
|
|||||||
editable_settings,
|
editable_settings,
|
||||||
database_settings,
|
database_settings,
|
||||||
&server_data_dir,
|
&server_data_dir,
|
||||||
runtime,
|
pools,
|
||||||
common_state::State::pools(common::resources::GameMode::Server),
|
|
||||||
)
|
)
|
||||||
.expect("Failed to create server instance!");
|
.expect("Failed to create server instance!");
|
||||||
|
|
||||||
|
@ -212,7 +212,6 @@ impl Server {
|
|||||||
editable_settings: EditableSettings,
|
editable_settings: EditableSettings,
|
||||||
database_settings: DatabaseSettings,
|
database_settings: DatabaseSettings,
|
||||||
data_dir: &std::path::Path,
|
data_dir: &std::path::Path,
|
||||||
runtime: Arc<Runtime>,
|
|
||||||
pools: common_state::Pools,
|
pools: common_state::Pools,
|
||||||
) -> Result<Self, Error> {
|
) -> Result<Self, Error> {
|
||||||
info!("Server data dir is: {}", data_dir.display());
|
info!("Server data dir is: {}", data_dir.display());
|
||||||
@ -251,13 +250,13 @@ impl Server {
|
|||||||
},
|
},
|
||||||
calendar: Some(settings.calendar_mode.calendar_now()),
|
calendar: Some(settings.calendar_mode.calendar_now()),
|
||||||
},
|
},
|
||||||
&pools.2,
|
&pools.rayon_pool,
|
||||||
);
|
);
|
||||||
#[cfg(not(feature = "worldgen"))]
|
#[cfg(not(feature = "worldgen"))]
|
||||||
let (world, index) = World::generate(settings.world_seed);
|
let (world, index) = World::generate(settings.world_seed);
|
||||||
|
|
||||||
#[cfg(feature = "worldgen")]
|
#[cfg(feature = "worldgen")]
|
||||||
let map = world.get_map_data(index.as_index_ref(), &pools.2);
|
let map = world.get_map_data(index.as_index_ref(), &pools.rayon_pool);
|
||||||
#[cfg(not(feature = "worldgen"))]
|
#[cfg(not(feature = "worldgen"))]
|
||||||
let map = WorldMapMsg {
|
let map = WorldMapMsg {
|
||||||
dimensions_lg: Vec2::zero(),
|
dimensions_lg: Vec2::zero(),
|
||||||
@ -270,6 +269,7 @@ impl Server {
|
|||||||
default_chunk: Arc::new(world.generate_oob_chunk()),
|
default_chunk: Arc::new(world.generate_oob_chunk()),
|
||||||
};
|
};
|
||||||
|
|
||||||
|
let runtime = Arc::clone(&pools.runtime);
|
||||||
let mut state = State::server(
|
let mut state = State::server(
|
||||||
pools,
|
pools,
|
||||||
world.sim().map_size_lg(),
|
world.sim().map_size_lg(),
|
||||||
|
@ -415,10 +415,12 @@ impl<'a> System<'a> for Sys {
|
|||||||
chunk
|
chunk
|
||||||
})
|
})
|
||||||
}).collect::<Vec<_>>();
|
}).collect::<Vec<_>>();
|
||||||
// Drop chunks in a background thread.
|
if !chunks_to_remove.is_empty() {
|
||||||
slow_jobs.spawn(&"CHUNK_DROP", async move {
|
// Drop chunks in a background thread.
|
||||||
drop(chunks_to_remove);
|
slow_jobs.spawn(&"CHUNK_DROP", async move {
|
||||||
});
|
drop(chunks_to_remove);
|
||||||
|
});
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -59,7 +59,6 @@ use i18n::LocalizationHandle;
|
|||||||
use std::path::PathBuf;
|
use std::path::PathBuf;
|
||||||
|
|
||||||
use std::sync::Arc;
|
use std::sync::Arc;
|
||||||
use tokio::runtime::Runtime;
|
|
||||||
|
|
||||||
/// A type used to store state that is shared between all play states.
|
/// A type used to store state that is shared between all play states.
|
||||||
pub struct GlobalState {
|
pub struct GlobalState {
|
||||||
@ -68,7 +67,6 @@ pub struct GlobalState {
|
|||||||
pub settings: Settings,
|
pub settings: Settings,
|
||||||
pub profile: Profile,
|
pub profile: Profile,
|
||||||
pub window: Window,
|
pub window: Window,
|
||||||
pub tokio_runtime: Arc<Runtime>,
|
|
||||||
#[cfg(feature = "egui-ui")]
|
#[cfg(feature = "egui-ui")]
|
||||||
pub egui_state: EguiState,
|
pub egui_state: EguiState,
|
||||||
pub lazy_init: scene::terrain::SpriteRenderContextLazy,
|
pub lazy_init: scene::terrain::SpriteRenderContextLazy,
|
||||||
|
@ -181,29 +181,6 @@ fn main() {
|
|||||||
default_hook(panic_info);
|
default_hook(panic_info);
|
||||||
}));
|
}));
|
||||||
|
|
||||||
// Setup tokio runtime
|
|
||||||
use common::consts::MIN_RECOMMENDED_TOKIO_THREADS;
|
|
||||||
use std::sync::{
|
|
||||||
atomic::{AtomicUsize, Ordering},
|
|
||||||
Arc,
|
|
||||||
};
|
|
||||||
use tokio::runtime::Builder;
|
|
||||||
|
|
||||||
// TODO: evaluate std::thread::available_concurrency as a num_cpus replacement
|
|
||||||
let cores = num_cpus::get();
|
|
||||||
let tokio_runtime = Arc::new(
|
|
||||||
Builder::new_multi_thread()
|
|
||||||
.enable_all()
|
|
||||||
.worker_threads((cores / 4).max(MIN_RECOMMENDED_TOKIO_THREADS))
|
|
||||||
.thread_name_fn(|| {
|
|
||||||
static ATOMIC_ID: AtomicUsize = AtomicUsize::new(0);
|
|
||||||
let id = ATOMIC_ID.fetch_add(1, Ordering::SeqCst);
|
|
||||||
format!("tokio-voxygen-{}", id)
|
|
||||||
})
|
|
||||||
.build()
|
|
||||||
.unwrap(),
|
|
||||||
);
|
|
||||||
|
|
||||||
#[cfg(feature = "hot-reloading")]
|
#[cfg(feature = "hot-reloading")]
|
||||||
assets::start_hot_reloading();
|
assets::start_hot_reloading();
|
||||||
|
|
||||||
@ -253,7 +230,7 @@ fn main() {
|
|||||||
|
|
||||||
// Create window
|
// Create window
|
||||||
use veloren_voxygen::{error::Error, render::RenderError};
|
use veloren_voxygen::{error::Error, render::RenderError};
|
||||||
let (mut window, event_loop) = match Window::new(&settings, &tokio_runtime) {
|
let (mut window, event_loop) = match Window::new(&settings) {
|
||||||
Ok(ok) => ok,
|
Ok(ok) => ok,
|
||||||
// Custom panic message when a graphics backend could not be found
|
// Custom panic message when a graphics backend could not be found
|
||||||
Err(Error::RenderError(RenderError::CouldNotFindAdapter)) => {
|
Err(Error::RenderError(RenderError::CouldNotFindAdapter)) => {
|
||||||
@ -290,7 +267,6 @@ fn main() {
|
|||||||
audio,
|
audio,
|
||||||
profile,
|
profile,
|
||||||
window,
|
window,
|
||||||
tokio_runtime,
|
|
||||||
#[cfg(feature = "egui-ui")]
|
#[cfg(feature = "egui-ui")]
|
||||||
egui_state,
|
egui_state,
|
||||||
lazy_init,
|
lazy_init,
|
||||||
|
@ -11,7 +11,6 @@ use std::{
|
|||||||
},
|
},
|
||||||
time::Duration,
|
time::Duration,
|
||||||
};
|
};
|
||||||
use tokio::runtime;
|
|
||||||
use tracing::{trace, warn};
|
use tracing::{trace, warn};
|
||||||
|
|
||||||
#[derive(Debug)]
|
#[derive(Debug)]
|
||||||
@ -47,7 +46,6 @@ impl ClientInit {
|
|||||||
connection_args: ConnectionArgs,
|
connection_args: ConnectionArgs,
|
||||||
username: String,
|
username: String,
|
||||||
password: String,
|
password: String,
|
||||||
runtime: Arc<runtime::Runtime>,
|
|
||||||
pools: common_state::Pools,
|
pools: common_state::Pools,
|
||||||
) -> Self {
|
) -> Self {
|
||||||
let (tx, rx) = unbounded();
|
let (tx, rx) = unbounded();
|
||||||
@ -55,9 +53,7 @@ impl ClientInit {
|
|||||||
let cancel = Arc::new(AtomicBool::new(false));
|
let cancel = Arc::new(AtomicBool::new(false));
|
||||||
let cancel2 = Arc::clone(&cancel);
|
let cancel2 = Arc::clone(&cancel);
|
||||||
|
|
||||||
let runtime2 = Arc::clone(&runtime);
|
pools.runtime.spawn(async move {
|
||||||
|
|
||||||
runtime.spawn(async move {
|
|
||||||
let trust_fn = |auth_server: &str| {
|
let trust_fn = |auth_server: &str| {
|
||||||
let _ = tx.send(Msg::IsAuthTrusted(auth_server.to_string()));
|
let _ = tx.send(Msg::IsAuthTrusted(auth_server.to_string()));
|
||||||
trust_rx
|
trust_rx
|
||||||
@ -76,7 +72,6 @@ impl ClientInit {
|
|||||||
let mut mismatched_server_info = None;
|
let mut mismatched_server_info = None;
|
||||||
match Client::new(
|
match Client::new(
|
||||||
connection_args.clone(),
|
connection_args.clone(),
|
||||||
Arc::clone(&runtime2),
|
|
||||||
&mut mismatched_server_info,
|
&mut mismatched_server_info,
|
||||||
pools.clone(),
|
pools.clone(),
|
||||||
&username,
|
&username,
|
||||||
@ -87,8 +82,7 @@ impl ClientInit {
|
|||||||
{
|
{
|
||||||
Ok(client) => {
|
Ok(client) => {
|
||||||
let _ = tx.send(Msg::Done(Ok(client)));
|
let _ = tx.send(Msg::Done(Ok(client)));
|
||||||
drop(pools);
|
tokio::task::block_in_place(move || drop(pools));
|
||||||
tokio::task::block_in_place(move || drop(runtime2));
|
|
||||||
return;
|
return;
|
||||||
},
|
},
|
||||||
Err(ClientError::NetworkErr(NetworkError::ConnectFailed(
|
Err(ClientError::NetworkErr(NetworkError::ConnectFailed(
|
||||||
@ -112,9 +106,8 @@ impl ClientInit {
|
|||||||
// address and all the attempts timed out.
|
// address and all the attempts timed out.
|
||||||
let _ = tx.send(Msg::Done(Err(last_err.unwrap_or(Error::ServerNotFound))));
|
let _ = tx.send(Msg::Done(Err(last_err.unwrap_or(Error::ServerNotFound))));
|
||||||
|
|
||||||
drop(pools);
|
|
||||||
// Safe drop runtime
|
// Safe drop runtime
|
||||||
tokio::task::block_in_place(move || drop(runtime2));
|
tokio::task::block_in_place(move || drop(pools));
|
||||||
});
|
});
|
||||||
|
|
||||||
ClientInit {
|
ClientInit {
|
||||||
|
@ -22,7 +22,7 @@ use common_base::span;
|
|||||||
use i18n::LocalizationHandle;
|
use i18n::LocalizationHandle;
|
||||||
use scene::Scene;
|
use scene::Scene;
|
||||||
use std::sync::Arc;
|
use std::sync::Arc;
|
||||||
use tokio::runtime;
|
use tokio::Builder;
|
||||||
use tracing::error;
|
use tracing::error;
|
||||||
use ui::{Event as MainMenuEvent, MainMenuUi};
|
use ui::{Event as MainMenuEvent, MainMenuUi};
|
||||||
|
|
||||||
@ -104,7 +104,7 @@ impl PlayState for MainMenuState {
|
|||||||
"".to_owned(),
|
"".to_owned(),
|
||||||
ConnectionArgs::Mpsc(14004),
|
ConnectionArgs::Mpsc(14004),
|
||||||
&mut self.init,
|
&mut self.init,
|
||||||
&global_state.tokio_runtime,
|
&pools.runtime,
|
||||||
&global_state.i18n,
|
&global_state.i18n,
|
||||||
Some(pools),
|
Some(pools),
|
||||||
);
|
);
|
||||||
@ -300,7 +300,6 @@ impl PlayState for MainMenuState {
|
|||||||
password,
|
password,
|
||||||
connection_args,
|
connection_args,
|
||||||
&mut self.init,
|
&mut self.init,
|
||||||
&global_state.tokio_runtime,
|
|
||||||
&global_state.i18n,
|
&global_state.i18n,
|
||||||
None,
|
None,
|
||||||
);
|
);
|
||||||
@ -331,7 +330,8 @@ impl PlayState for MainMenuState {
|
|||||||
},
|
},
|
||||||
#[cfg(feature = "singleplayer")]
|
#[cfg(feature = "singleplayer")]
|
||||||
MainMenuEvent::StartSingleplayer => {
|
MainMenuEvent::StartSingleplayer => {
|
||||||
let singleplayer = Singleplayer::new(&global_state.tokio_runtime);
|
|
||||||
|
let singleplayer = Singleplayer::new();
|
||||||
|
|
||||||
global_state.singleplayer = Some(singleplayer);
|
global_state.singleplayer = Some(singleplayer);
|
||||||
},
|
},
|
||||||
@ -500,7 +500,6 @@ fn attempt_login(
|
|||||||
password: String,
|
password: String,
|
||||||
connection_args: ConnectionArgs,
|
connection_args: ConnectionArgs,
|
||||||
init: &mut InitState,
|
init: &mut InitState,
|
||||||
runtime: &Arc<runtime::Runtime>,
|
|
||||||
localized_strings: &LocalizationHandle,
|
localized_strings: &LocalizationHandle,
|
||||||
pools: Option<common_state::Pools>,
|
pools: Option<common_state::Pools>,
|
||||||
) {
|
) {
|
||||||
@ -533,7 +532,12 @@ fn attempt_login(
|
|||||||
username,
|
username,
|
||||||
password,
|
password,
|
||||||
Arc::clone(runtime),
|
Arc::clone(runtime),
|
||||||
pools.unwrap_or_else(|| common_state::State::pools(common::resources::GameMode::Client)),
|
pools.unwrap_or_else(|| {
|
||||||
|
common_state::State::pools(
|
||||||
|
common::resources::GameMode::Client,
|
||||||
|
tokio::runtime::Builder::new_multi_thread(),
|
||||||
|
)
|
||||||
|
}),
|
||||||
));
|
));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -198,7 +198,6 @@ impl Renderer {
|
|||||||
pub fn new(
|
pub fn new(
|
||||||
window: &winit::window::Window,
|
window: &winit::window::Window,
|
||||||
mode: RenderMode,
|
mode: RenderMode,
|
||||||
runtime: &tokio::runtime::Runtime,
|
|
||||||
) -> Result<Self, RenderError> {
|
) -> Result<Self, RenderError> {
|
||||||
let (pipeline_modes, mut other_modes) = mode.split();
|
let (pipeline_modes, mut other_modes) = mode.split();
|
||||||
// Enable seamless cubemaps globally, where available--they are essentially a
|
// Enable seamless cubemaps globally, where available--they are essentially a
|
||||||
@ -312,6 +311,9 @@ impl Renderer {
|
|||||||
path
|
path
|
||||||
});
|
});
|
||||||
|
|
||||||
|
let runtime = runtime::Builder::new_current_thread().build()
|
||||||
|
.expect("Failed to create single-threaded tokio runtime (for renderer initialization).");
|
||||||
|
|
||||||
let (device, queue) = runtime.block_on(adapter.request_device(
|
let (device, queue) = runtime.block_on(adapter.request_device(
|
||||||
&wgpu::DeviceDescriptor {
|
&wgpu::DeviceDescriptor {
|
||||||
// TODO
|
// TODO
|
||||||
|
@ -12,7 +12,6 @@ use std::{
|
|||||||
thread::{self, JoinHandle},
|
thread::{self, JoinHandle},
|
||||||
time::Duration,
|
time::Duration,
|
||||||
};
|
};
|
||||||
use tokio::runtime::Runtime;
|
|
||||||
use tracing::{error, info, trace, warn};
|
use tracing::{error, info, trace, warn};
|
||||||
|
|
||||||
const TPS: u64 = 30;
|
const TPS: u64 = 30;
|
||||||
@ -30,7 +29,7 @@ pub struct Singleplayer {
|
|||||||
}
|
}
|
||||||
|
|
||||||
impl Singleplayer {
|
impl Singleplayer {
|
||||||
pub fn new(runtime: &Arc<Runtime>) -> Self {
|
pub fn new() -> Self {
|
||||||
let (stop_server_s, stop_server_r) = unbounded();
|
let (stop_server_s, stop_server_r) = unbounded();
|
||||||
|
|
||||||
// Determine folder to save server data in
|
// Determine folder to save server data in
|
||||||
@ -100,18 +99,19 @@ impl Singleplayer {
|
|||||||
let (result_sender, result_receiver) = bounded(1);
|
let (result_sender, result_receiver) = bounded(1);
|
||||||
|
|
||||||
let builder = thread::Builder::new().name("singleplayer-server-thread".into());
|
let builder = thread::Builder::new().name("singleplayer-server-thread".into());
|
||||||
let runtime = Arc::clone(runtime);
|
|
||||||
let thread = builder
|
let thread = builder
|
||||||
.spawn(move || {
|
.spawn(move || {
|
||||||
trace!("starting singleplayer server thread");
|
trace!("starting singleplayer server thread");
|
||||||
|
|
||||||
let pools = common_state::State::pools(common::resources::GameMode::Singleplayer);
|
let pools = common_state::State::pools(
|
||||||
|
common::resources::GameMode::Singleplayer,
|
||||||
|
tokio::runtime::Builder::new_multi_thread(),
|
||||||
|
);
|
||||||
let (server, init_result) = match Server::new(
|
let (server, init_result) = match Server::new(
|
||||||
settings2,
|
settings2,
|
||||||
editable_settings,
|
editable_settings,
|
||||||
database_settings,
|
database_settings,
|
||||||
&server_data_dir,
|
&server_data_dir,
|
||||||
runtime,
|
|
||||||
pools.clone(),
|
pools.clone(),
|
||||||
) {
|
) {
|
||||||
Ok(server) => (Some(server), Ok(pools)),
|
Ok(server) => (Some(server), Ok(pools)),
|
||||||
|
@ -411,10 +411,7 @@ pub struct Window {
|
|||||||
}
|
}
|
||||||
|
|
||||||
impl Window {
|
impl Window {
|
||||||
pub fn new(
|
pub fn new(settings: &Settings) -> Result<(Window, EventLoop), Error> {
|
||||||
settings: &Settings,
|
|
||||||
runtime: &tokio::runtime::Runtime,
|
|
||||||
) -> Result<(Window, EventLoop), Error> {
|
|
||||||
let event_loop = EventLoop::new();
|
let event_loop = EventLoop::new();
|
||||||
|
|
||||||
let size = settings.graphics.window_size;
|
let size = settings.graphics.window_size;
|
||||||
@ -434,7 +431,7 @@ impl Window {
|
|||||||
|
|
||||||
let window = win_builder.build(&event_loop).unwrap();
|
let window = win_builder.build(&event_loop).unwrap();
|
||||||
|
|
||||||
let renderer = Renderer::new(&window, settings.graphics.render_mode.clone(), runtime)?;
|
let renderer = Renderer::new(&window, settings.graphics.render_mode.clone())?;
|
||||||
|
|
||||||
let keypress_map = HashMap::new();
|
let keypress_map = HashMap::new();
|
||||||
|
|
||||||
|
Loading…
Reference in New Issue
Block a user