You signed in with another tab or window. Reload to refresh your session.You signed out in another tab or window. Reload to refresh your session.You switched accounts on another tab or window. Reload to refresh your session.Dismiss alert
I believe there is a problem with the design of nb as it only allows for "one level" of asynchronicity, which is glossed over via the stubbed out definitions for the sample APIs in the documentation (such as serial_send and serial_receive).
If you were to actually flesh out those examples into real-world code, a problem arises wherein a nb function can really only decide that it would block once and only once. The moment stateful work commences that is predicated on the values that were passed in to the function, it becomes impossible to "early out" for the remainder of the function while still supporting a level of abstraction in which the function itself does not need to know anything about the caller.
For example, take the following pseudocode:
fnsend_something(&mutself,bytes:&[u8]) -> nb::Result<(), !>{let result = self.already_sending_something.compare_exchange(false,true,Ordering:..);if result != Ok(false){// Awesome, we avoided blocking when the sender was busyreturnErr(nb::Error::WouldBlock);}for i in0..bytes.len(){// Start sending by loading this byte into the send registerself.send_register.load(bytes[i]);// We could poll until the byte has been sent// while !self.send_register.is_empty() {};// But why don't we avoid blocking instead?if !self.send_register.is_empty(){return nb::Error::WouldBlock;}}returnOk(());}
This code can be adapted to store i internally (saving state) so that the next time it is called after self.send_register.is_empty() returns false, it can pick up where it left off instead of starting over from zero, no problem. But how does it know that it is being called again by the same line of code/state?
The text was updated successfully, but these errors were encountered:
I understand the highlighted problem but for the example above I would say that to me "send something" sounds too "blocking", and then a non-blocking implementation gets weird.
I would say that in this example the "forgetfulness of nb" could be passed on to the API. i.e.:
Whoever processes something asynchronously could have a queue of uncompleted work. Then I would split the method something like enqueue_something_for_send(). and process_send_queue() (or hopefully some better names than that). This would break the dependency to the function argument.
To put this into some pseudocode, given an imaginary queue:
fnenqueue_for_send(&mutself,bytes:&[u8]){// you may need to return some error in case the queue is full or something like thatself.send_queue.push(bytes);}fnprocess_send_queue(&mutself)-> nb::Result<(), !>{let result = self.already_sending_something.compare_exchange(false,true,Ordering:..);if result != Ok(false){// Awesome, we avoided blocking when the sender was busyreturnErr(nb::Error::WouldBlock);}let next_byte = self.send_queue.pop();ifletOk(byte) = next_byte {self.send_register.load(byte);returnErr(nb::Error::WouldBlock);}else{returnOk(());// queue was empty}}
This could be further improved to handle situations where the same byte should be retried because an error occurred or avoid the last process_send_queue() call, etc.
This would enable scenarios where two different loops enqueue and actually send data, as well as scenarios where one thread does it all (enqueue once, block on process_send_queue())
If you are interested in checking if a particular data piece was already sent, the code could also be extended to return a token to the element inserted in the send queue. Then the user can check later on, or even provide the token with a function to also process the queue but I think the whole thing starts getting more complicated.
I believe there is a problem with the design of
nb
as it only allows for "one level" of asynchronicity, which is glossed over via the stubbed out definitions for the sample APIs in the documentation (such asserial_send
andserial_receive
).If you were to actually flesh out those examples into real-world code, a problem arises wherein a
nb
function can really only decide that it would block once and only once. The moment stateful work commences that is predicated on the values that were passed in to the function, it becomes impossible to "early out" for the remainder of the function while still supporting a level of abstraction in which the function itself does not need to know anything about the caller.For example, take the following pseudocode:
This code can be adapted to store
i
internally (saving state) so that the next time it is called afterself.send_register.is_empty()
returns false, it can pick up where it left off instead of starting over from zero, no problem. But how does it know that it is being called again by the same line of code/state?The text was updated successfully, but these errors were encountered: