Triggering false clicks when two avatars have control permission (root prim and child prim).
tracked
Zack Ormidale
I found a bug that causes false clicks on the avatar when there is a passenger seat with control permissions. In other words, the avatar sitting on the root prim has a script to allow controls and the avatar sitting on the prim child has another script on the prim child that also allows controls.
This error can only be caught if the script depends on correctly reading button presses. When both avatars are sitting, and the root avatar holds down a key, it will fire even when it is just pressed. It takes 10 to 20 seconds for this bug to start, and if there is more load on the script it can trigger very quickly.
The click receiver is defined as a "first touch" limiter, but the avatar will hold the key in the "controls section":
integer start = level & edge;
if(start & CONTROL_FWD)
This shouldn't let any command pass without the avatar releasing the key and pressing it again, but in this bug it triggers false clicks with the key held down.
I wrote a script to demonstrate this bug in practice. There are two scripts, one for root and one for child prim:
Connect 2 prims for seat.
Each Prim receives the respective scripts below that request control permission. Sit with both avatars. Use the main avatar in root prim: press forward, hold for up to 20 seconds and see false clicks being generated:
Scripts that catch the bug:
------------------------------------------------------------
//Root Prim:
integer i;
default
{
state_entry()
{
llSetTimerEvent(0.2);
}
changed(integer change)
{
if(change & CHANGED_LINK)
{
key avatar = llAvatarOnSitTarget();
if(avatar != NULL_KEY)
{
llRequestPermissions(avatar, PERMISSION_TAKE_CONTROLS);
}
else
{
llReleaseControls();
llResetScript();
}
}
}
run_time_permissions(integer perm)
{
if(PERMISSION_TAKE_CONTROLS & perm)
{
llTakeControls(CONTROL_RIGHT | CONTROL_LEFT | CONTROL_BACK | CONTROL_FWD, TRUE, 0);
}
}
control(key id, integer level, integer edge)
{
integer start = level & edge;
if(start & CONTROL_FWD)//Only one click at a time
{
i++;
llSay(0,"Click = "+(string)i);
}
}
timer()
{
//just to increase the use of the script in this bug test.
integer read_timer;
read_timer++;
}
}
//End script 1
------------------------------------------------------------------------
-------------------------------------------------------------------------
//Child Prim:
default
{
changed(integer change)
{
if(change & CHANGED_LINK)
{
key avatar = llAvatarOnSitTarget();
if(avatar != NULL_KEY)
{
llRequestPermissions(avatar, PERMISSION_TAKE_CONTROLS);
}
else
{
llReleaseControls();
llResetScript();
}
}
}
run_time_permissions(integer perm)
{
if(PERMISSION_TAKE_CONTROLS & perm)
{
llTakeControls(CONTROL_RIGHT | CONTROL_LEFT | CONTROL_BACK | CONTROL_FWD, TRUE, 0);
}
}
control(key id, integer level, integer edge)
{
//Passenger. Nothing defined in this test. Permission only.
}
}
//End Script 2
-------------------------------------------------------------------------
This is a translator. I would like someone from the native language to send this for greater clarity, but unfortunately the error happened to me.
I hope that Linden resolves this very calmly and cautiously, as I know that fixing a bug can generate other bugs. I'm not in a hurry or urgent. At the moment I could deviate from this type of script, but I would like this to be listed as bugs to be fixed.
Thanks!
Log In
Infrared Starlight
I have independently come across this issue, which also breaks anything involving passengers that can fire out of vehicles for an otherwise armed vehicle. When the passengers fire (like an APC, for example), so too does the gun for the driver fire. Not only is it generating input from them for the root prim, it is also the driver's key, so there is no way to filter it out.
This also happens for unrelated control events.
Root:
default
{
state_entry() {
llSitTarget(<0,0,.5>,ZERO_ROTATION);
}
changed(integer change) {
if(change&CHANGED_LINK) {
if(llAvatarOnSitTarget() != NULL_KEY) {
llRequestPermissions(llAvatarOnSitTarget(),PERMISSION_TAKE_CONTROLS);
}
}
}
run_time_permissions(integer perm) {
if(perm) {
llTakeControls(CONTROL_ML_LBUTTON,TRUE,TRUE);
}
}
control(key k, integer h, integer c) {
llOwnerSay((string)k);
}
}
Child:
default
{
state_entry() {
llSitTarget(<0,0,.5>,ZERO_ROTATION);
}
changed(integer change) {
if(change&CHANGED_LINK) {
if(llAvatarOnLinkSitTarget(LINK_THIS)) { llRequestPermissions(llAvatarOnLinkSitTarget(LINK_THIS),PERMISSION_TAKE_CONTROLS);
}
}
}
run_time_permissions(integer perm) {
if(perm) {
llTakeControls(CONTROL_DOWN,TRUE,TRUE);
}
}
control(key k, integer h, integer c) {
if(h&c) {
llUnSit(llAvatarOnLinkSitTarget(LINK_THIS));
}
}
}
Wear a popgun on the child prim. The root prim also receives CONTROL_ML_LBUTTON originating from the user on the root prim, which is certainly a bug.
Maestro Linden
tracked
Maestro Linden
Hi Zack Ormidale thanks for the report. I can reproduce the issue.
I took your scripts and added some debug chat to them (shown below). When I perform these steps, I can reproduce the bug:
- UserA: sit on the root prim. The root prim script automatically grabs controls from UserA.
- UserA: press and hold "W" for a while.
- Observe that a single control event triggers from the script: "Click = 1 from e1a51880-d7b5-4c00-800d-91664f5b84c0"
- UserA: release "W"
- UserB: sit on the child prim. The child prim script automatically grabs controls from UserB.
- UserA: press and hold "W"
- Observe that a single control event triggers from the script: "Click = 2 from e1a51880-d7b5-4c00-800d-91664f5b84c0"(so far, as expected)
- UserB: press the WASD keys several times
Expected results:
In step 8, there should be no additional "Click" messages from the root prim script - this message should only trigger when UserA first presses the button. Indeed, the root prim's script shouldn't get any control() events from any agent other than UserA, since it has only taken UserA's controls.
Actual results:
In step 8, there is an additional "Click" message, due to a control event in the root prim, which is falsely attributed to UserA:
root: Click = 2 from e1a51880-d7b5-4c00-800d-91664f5b84c0
root: Click = 3 from e1a51880-d7b5-4c00-800d-91664f5b84c0
root: Click = 4 from e1a51880-d7b5-4c00-800d-91664f5b84c0
root: Click = 5 from e1a51880-d7b5-4c00-800d-91664f5b84c0
// Script 1 - root prim
integer i;
default
{
state_entry()
{
llSetTimerEvent(0.2);
// Add a sit target so that llAvatarOnSitTarget() will work properly
llSitTarget(<0,0,0.5>, ZERO_ROTATION);
// label prim
llSetObjectName("root");
llSetText((string)llGetLinkNumber(), <1,1,1>, 1);
}
changed(integer change)
{
if(change & CHANGED_LINK)
{
key avatar = llAvatarOnSitTarget();
if(avatar != NULL_KEY)
{
llRequestPermissions(avatar, PERMISSION_TAKE_CONTROLS);
llSay(0, "Requested PERMISSION_TAKE_CONTROLS from " + (string)avatar);
}
else
{
llSay(0, "Releasing controls and resetting script");
llReleaseControls();
llResetScript();
}
}
}
run_time_permissions(integer perm)
{
if(PERMISSION_TAKE_CONTROLS & perm)
{
llTakeControls(CONTROL_RIGHT | CONTROL_LEFT | CONTROL_BACK | CONTROL_FWD, TRUE, 0);
llSay(0, "Taking controls on " + (string)llGetPermissionsKey());
}
}
control(key id, integer level, integer edge)
{
integer start = level & edge;
if(start & CONTROL_FWD)//Only one click at a time
{
i++;
llSay(0,"Click = "+(string)i + " from " + (string)id);
}
}
timer()
{
//just to increase the use of the script in this bug test.
integer read_timer;
read_timer++;
}
}
// script 2 - child prim
default
{
state_entry()
{
// Add a sit target so that llAvatarOnSitTarget() will work properly
llSitTarget(<0,0,0.5>, ZERO_ROTATION);
// label prim
llSetObjectName("child");
llSetText((string)llGetLinkNumber(), <1,1,1>, 1);
}
changed(integer change)
{
if(change & CHANGED_LINK)
{
key avatar = llAvatarOnSitTarget();
if(avatar != NULL_KEY)
{
llSay(0, "Requested PERMISSION_TAKE_CONTROLS from " + (string)avatar);
llRequestPermissions(avatar, PERMISSION_TAKE_CONTROLS);
}
else
{
llSay(0, "Releasing controls and resetting script");
llReleaseControls();
llResetScript();
}
}
}
run_time_permissions(integer perm)
{
if(PERMISSION_TAKE_CONTROLS & perm)
{
llTakeControls(CONTROL_RIGHT | CONTROL_LEFT | CONTROL_BACK | CONTROL_FWD, TRUE, 0);
llSay(0, "Taking controls on " + (string)llGetPermissionsKey());
}
}
control(key id, integer level, integer edge)
{
//Passenger. Nothing defined in this test. Permission only.
}
}
Zack Ormidale
Maestro Linden Thanks for the answer. I'm glad someone translated this into more technical terms. Whenever I can I update myself on the topic.
R
Ruiko Resident
Maestro Linden
Good day to you.
As the specification of llTakeControls and cotrol event, is the behaviour in linked prims with each prim's script different from in one prim with two scripts?
Is this not the same issue as SVC-3187?
"There are some bugs when you put two scripts in the same prim and call LlTakeControls(), the id may not be the intended one. See SVC-3187."
Maestro Linden
Ruiko Resident Yes it definitely seems related to SVC-3187. Though reading the original report in that issue, there's no mention of the wrong id in that report (the repro steps are vague, but it seems like a single agent would be generating control events in both scripts).
KingOfNorway Oxidor
Maestro Linden, I am having a similar problem. I have a vehicle that I would like to be controled by avatar A -OR- avatar B. In my scripts, (which are very complicated and involved with several levels of vehicle control) I have a hud that allows the driver to switch from Avatar A to Avatar B and vice versa.
The sequence is as follows:
1) Driver A automatically takes control when they sit and all the controls work as designed.
2) Driver B sits
All the controls (key start, key held, and key release) work fine for avatar A and avatar B cannot affect the vehicle. This is the expected behavior
3) A button is pressed to switch control. This triggers an
llReleaseControl() for A
and llTakeControl for B.
4) Now both Avatar A and Avatar B seem to be able to still control the vehicle even though control should have been removed for A -
- AND -
the control keys behave the same AND erratically for either A or B - When a key is held down, I should get an unbroken stream of "keydown" messages
- BUT-
the script registers random "keypresses" and "keyreleases" in the midsts of the "keydown" stream. This, of course disrupts lots of things that are going on when keys are held down or released.
5) If the controls are then released from Avatar B and reassigned to Avatar A, it all seems to work well for A, and B is locked out as would be expected.
I was not able to view SVC-3187 to see if this is the same problem. Where would I find it? Is it on GitHub? clicking on the link in the LSL wiki takes me to GitHub.
KingOfNorway Oxidor
I did some further testing, because in a very simple script with just control assignments, it seems to work as designed. Howerver if I add a second script that also assigns controls, and I only reassign the avatar controller to on of those scripts, then I begin to see the flakey behavior described in the message above. My product is very complicated with a dozen or so scripts and it is likely that this is the case since I use key control for a number of different tasks. I will search it out and see if I can find more detailed conflicts.
KingOfNorway Oxidor
I followed up and discovered that I was indeed calling for llTakeControls in another script in the product. Even when I silenced that script call, then the one remaining call to llTakeControls functioned eratically when assigned to avatar B. It seems that trying to assign control to the secondary avatar is not working properly .
KingOfNorway Oxidor
After further searching in the product, I eliminated all the llTakeControls in various scripts except the major one of interest and discovered that if there is ONLY one instance of it, it seems to work fine. Whew! perhaps the fix needs to somehow segregate the llTakeControls calls so they onluy work and be turned on and off for only one script. Hope this is helpful.