Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Inline optimization with run still makes calls to modify #33

Open
milesfrain opened this issue Apr 25, 2020 · 1 comment
Open

Inline optimization with run still makes calls to modify #33

milesfrain opened this issue Apr 25, 2020 · 1 comment
Labels
purs-0.15 A reminder to address this issue or merge this PR before we release PureScript v0.15.0 type: breaking change A change that requires a major version bump.

Comments

@milesfrain
Copy link

Originally reported here.

There's an unexpected call to Control_Monad_ST_Internal.modify in the JS output of the following function:

simulate :: Number -> Number -> Int -> Number
simulate x0 v0 time =
  run do
    ref <- new { x: x0, v: v0 }
    for 0 (time * 1000) \_ ->
      modify
        ( \o ->
            { v: o.v - 9.81 * 0.001
            , x: o.x + o.v * 0.001
            }
        )
        ref
    final <- read ref
    pure final.x
var ref = { value: { x: x0, v: v0 } };                
 
Control_Monad_ST_Internal["for"](0)(time * 1000 | 0)(function (v) {                
  return Control_Monad_ST_Internal.modify(function (o) {                  
    return {                    
      v: o.v - 9.81 * 1.0e-3,                      
      x: o.x + o.v * 1.0e-3                      
    };                    
  })(ref);                  
})();

The original version of the book shows a more optimized JS output that does not make a call to modify:

var ref = { x: x0, v: v0 };     
                                  
Control_Monad_Eff.forE(0)(time * 1000 | 0)(function (i) {    
  return function __do() {    
    ref = (function (o) {    
      return {                                                                         
        v: o.v - 9.81 * 1.0e-3,                                               
        x: o.x + o.v * 1.0e-3    
      };    
    })(ref);    
    return Prelude.unit;    
  };    
})();    
                                       
return ref.x;
@JordanMartinez JordanMartinez added the type: breaking change A change that requires a major version bump. label Nov 16, 2021
@JordanMartinez JordanMartinez added the purs-0.15 A reminder to address this issue or merge this PR before we release PureScript v0.15.0 label Dec 4, 2021
@sjpgarcia
Copy link

sjpgarcia commented Dec 30, 2021

Just to add my notes on this. This is what simulate looks like w/o inlining, plus the extra write for comparison:

var simulate = function (x0) {
    return function (v0) {
        return function (time) {
            return Control_Monad_ST_Internal.run(function __do() {
                var ref = Control_Monad_ST_Internal["new"]({
                    x: x0,
                    v: v0
                })();
                Control_Monad_ST_Internal["for"](0)(time * 1000 | 0)(function (v) {
                    return Control_Monad_ST_Internal.modify(function (o) {
                        return {
                            v: o.v - 9.81 * 1.0e-3,
                            x: o.x + o.v * 1.0e-3
                        };
                    })(ref);
                })();
                var $$final = Control_Monad_ST_Internal.read(ref)();
                Control_Monad_ST_Internal.write({
                    x: -1.0,
                    v: -1.0
                })(ref)();
                return $$final.x;
            });
        };
    };
};

What's interesting here is that all of the foreign functions (new, for, read, write) have an extra abstraction, while modify doesn't, because it's defined in PS. The ST optimizer in its current (unchanged) state accounts for when modify was still a foreign function, thus explaining the failures. I suppose the reasonable action would be to make a special case in the optimizer for modify.
This can be resolved by having MagicDo account for the code these utilities generate.

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
purs-0.15 A reminder to address this issue or merge this PR before we release PureScript v0.15.0 type: breaking change A change that requires a major version bump.
Projects
None yet
Development

No branches or pull requests

3 participants