Controller
This section introduces the APIs available in the Controller
.
fpga_state
Retrieve the state of the FPGA.
Before using this, you need to enable state retrieval with ReadsFPGAState
.
use autd3::prelude::*;
#[allow(unused_variables)]
fn main() -> Result<(), Box<dyn std::error::Error>> {
let mut autd = Controller::open([AUTD3::default()], autd3::link::Nop::new())?;
autd.send(ReadsFPGAState::new(|_dev| true))?;
let info = autd.fpga_state()?;
Ok(())
}
#include<autd3.hpp>
#include<autd3/link/nop.hpp>
int main() {
using namespace autd3;
auto autd =
Controller::open({AUTD3{}}, link::Nop{});
autd.send(ReadsFPGAState([](const auto&) { return true; }));
const auto info = autd.fpga_state();
return 0; }
using AUTD3Sharp;
using AUTD3Sharp.Link;
using AUTD3Sharp.Utils;
using var autd = Controller.Open([new AUTD3()], new Nop());
autd.Send(new ReadsFPGAState(_ => true));
var info = autd.FPGAState();
from pyautd3 import Controller, AUTD3, ReadsFPGAState
from pyautd3.link.nop import Nop
autd = Controller.open([AUTD3()], Nop())
autd.send(ReadsFPGAState(lambda _: True))
info = autd.fpga_state()
The argument of the ReadsFPGAState
constructor is Fn(&Device) -> bool
, which specifies whether to enable state retrieval for each device.
fpga_state
returns None
for devices that are not enabled.
The following information can currently be obtained as the state of the FPGA:
is_thermal_assert
: Whether the temperature sensor for fan control is assertedcurrent_mod_segment
: Current Modulation Segmentcurrent_stm_segment
: Current FociSTM/GainSTM Segmentcurrent_gain_segment
: Current Gain Segmentis_gain_mode
: Whether Gain is currently being usedis_stm_mode
: Whether FociSTM/GainSTM is currently being used
send
Send data to the device.
Data can be sent either individually or two at a time.
group_send
Using the group_send
function, you can group devices.
use autd3::prelude::*;
use autd3::gain::IntoBoxedGain;
use std::collections::HashMap;
fn main() -> Result<(), Box<dyn std::error::Error>> {
let mut autd = Controller::open(
[AUTD3::default(), AUTD3::default()],
autd3::link::Nop::new(),
)?;
let x = 0.;
let y = 0.;
let z = 0.;
autd.group_send(
|dev| match dev.idx() {
0 => Some("focus"),
1 => Some("null"),
_ => None,
},
HashMap::from([
(
"focus",
Focus {
pos: Point3::new(x, y, z),
option: Default::default(),
}
.into_boxed(),
),
("null", Null {}.into_boxed()),
]),
)?;
Ok(())
}
#include<chrono>
#include<autd3.hpp>
#include<autd3/link/nop.hpp>
int main() {
using namespace autd3;
auto autd =
Controller::open({AUTD3{}}, link::Nop{});
const auto x = 0.0;
const auto y = 0.0;
const auto z = 0.0;
autd.group_send(
[](const Device& dev) -> std::optional<const char*> {
if (dev.idx() == 0) {
return "null";
} else if (dev.idx() == 1) {
return "focus";
} else {
return std::nullopt;
}
},
std::unordered_map<const char*, std::shared_ptr<driver::Datagram>>{
{"focus",
std::make_shared<Focus>(Focus(Point3(x, y, z), FocusOption{}))},
{"null", std::make_shared<Null>()}});
return 0; }
using AUTD3Sharp;
using AUTD3Sharp.Link;
using AUTD3Sharp.Gain;
using AUTD3Sharp.Modulation;
using AUTD3Sharp.Utils;
using var autd = Controller.Open([new AUTD3()], new Nop());
var x = 0.0f;
var y = 0.0f;
var z = 0.0f;
autd.GroupSend(dev =>
{
return dev.Idx() switch
{
0 => "null",
1 => "focus",
_ => null
};
},
new GroupDictionary {
{ "null", new Null() },
{ "focus", new Focus(pos: new Point3(x, y, z), option: new FocusOption()) }
}
);
from pyautd3 import AUTD3, Controller, Device
from pyautd3.gain import Focus, FocusOption, Null
from pyautd3.link.nop import Nop
autd = Controller.open([AUTD3()], Nop())
x = 0.0
y = 0.0
z = 0.0
def key_map(dev: Device) -> str | None:
if dev.idx == 0:
return "null"
if dev.idx == 1:
return "focus"
return None
autd.group_send(
key_map=key_map,
data_map={"null": Null(), "focus": Focus(pos=[x, y, z], option=FocusOption())},
)
Unlike gain::Group
, you can use any data that can be sent with the usual send
.
However, you can only group by device.
NOTE: In this sample, strings are used as keys, but you can use anything that can be used as a key for
HashMap
.
sender
You can specify settings for sending via sender
.
use std::time::Duration;
use autd3::prelude::*;
fn main() -> Result<(), Box<dyn std::error::Error>> {
let mut autd = Controller::open([AUTD3::default()], autd3::link::Nop::new())?;
let mut sender = autd.sender(SenderOption {
send_interval: Duration::from_millis(1),
receive_interval: Duration::from_millis(1),
timeout: None,
parallel: ParallelMode::Auto,
sleeper: SpinSleeper::default(),
});
let d = Null {};
sender.send(d)?;
Ok(())
}
#include<chrono>
#include<autd3.hpp>
#include<autd3/link/nop.hpp>
int main() {
using namespace autd3;
auto autd = Controller::open({AUTD3{}}, link::Nop{});
auto sender = autd.sender(SenderOption{
.send_interval = std::chrono::milliseconds(1),
.receive_interval = std::chrono::milliseconds(1),
.timeout = std::nullopt,
.parallel = ParallelMode::Auto,
.sleeper = SpinSleeper(),
});
const Null d;
sender.send(d);
return 0; }
using AUTD3Sharp;
using AUTD3Sharp.Link;
using AUTD3Sharp.Gain;
using AUTD3Sharp.Utils;
using var autd = Controller.Open([new AUTD3()], new Nop());
var sender = autd.Sender(
new SenderOption
{
SendInterval = Duration.FromMillis(1),
ReceiveInterval = Duration.FromMillis(1),
Timeout = null,
Parallel = ParallelMode.Auto,
Sleeper = new SpinSleeper()
}
);
var d = new Null();
sender.Send(d);
from pyautd3 import AUTD3, Controller, Duration, Null, SenderOption, SpinSleeper, ParallelMode
from pyautd3.link.nop import Nop
autd = Controller.open([AUTD3()], Nop())
sender = autd.sender(
SenderOption(
send_interval=Duration.from_millis(1),
receive_interval=Duration.from_millis(1),
timeout=None,
parallel=ParallelMode.Auto,
sleeper=SpinSleeper(),
)
)
d = Null()
sender.send(d)
Here,
send_interval
: Send intervalreceive_interval
: Receive intervaltimeout
: Timeout duration. See About Timeout for detailsparallel
: Parallel computation mode. See About Parallel Computation for detailssleeper
: Structure to adjust send/receive intervalsSpinSleeper
: Usesspin_sleep
StdSleeper
: Usesstd::thread::sleep
WaitableSleeper
: (Windows only) UsesWaitable Timer
and he default values are as above.
Note that Controller::send
and Controller::group_send
are equivalent to Sender::send
and Sender::group_send
with the default SenderOption
.
About Timeout
If the timeout value is
- greater than 0, the
send
function waits until the sent data is processed by the device or the specified timeout duration elapses. If it cannot confirm that the sent data was processed by the device, it returns an error. - 0, the
send
function does not check whether the sent data was processed by the device.
If you want to ensure that the data is sent, it is recommended to set this to an appropriate value.
If timeout is not specified in SenderOption
, the default values for each data are used as shown below.
Timeout Value | |
---|---|
Clear /GPIOOutputs /ForceFan /PhaseCorrection /PulseWidthEncoder /ReadsFPGAState /SwapSegment /Silencer /Synchronize /FociSTM /GainSTM /Modulation | |
Gain |
When sending multiple data at once, the maximum timeout value of each data is used.
About Parallel Computation
Internal calculations for each data can be executed in parallel on a per-device basis.
Specifying ParallelMode::On
enables parallel computation, and ParallelMode::Off
disables it.
In the case of ParallelMode::Auto
, parallel computation is enabled if the number of enabled devices exceeds the parallel computation threshold value for each data as shown below.
Parallel Computation Threshold Value | |
---|---|
Clear /GPIOOutputs /ForceFan /PhaseCorrection /ReadsFPGAState /SwapSegment /Silencer /Synchronize /FociSTM (less than 4000 foci)/Modulation | 18446744073709551615 |
PulseWidthEncoder /FociSTM (4000 foci or more)// GainSTM /Gain | 4 |