Changeset 5404 in webkit


Ignore:
Timestamp:
Nov 6, 2003, 9:12:28 AM (21 years ago)
Author:
mjs
Message:

Reviewed by Darin.

  • fixed 3475366 - 4.5% of time spent making html event listeners on slow intel page.

6% speedup on intel page, 1% speedup on cvs-base PLT.

  • khtml/ecma/kjs_events.cpp: (JSEventListener::JSEventListener): Don't add self to hashtable if imp is null (which can now happen in the lazy listener case). (JSEventListener::~JSEventListener): Ditto on removing. (JSEventListener::listenerObj): Made this virtual. (JSLazyEventListener::JSLazyEventListener): New constructor. (JSLazyEventListener::handleEvent): call parseCode, then superclass if it appeared to succeed. (JSLazyEventListener::listenerObj): call parseCode, then superclass. (JSLazyEventListener::parseCode): (KJS::getNodeEventListener): Check for null listenerObjImp in case of lazy listener that failed to parse.
  • khtml/ecma/kjs_dom.cpp: (DOMNode::getListener): Ditto.
  • khtml/ecma/kjs_html.cpp: (Image::getValueProperty): Ditto.
  • khtml/ecma/kjs_events.h: (KJS::JSEventListener::listenerObjImp): call listenerObj() virtual method and get imp from the result.
  • khtml/ecma/kjs_proxy.cpp: (KJSProxyImpl::createHTMLEventHandler): Don't parse the code here, make a lazy listener.
  • khtml/ecma/kjs_window.cpp: (Window::getJSLazyEventListener): make a new JSLazyEventListener - no need to check the listeners hashtable cause a brand new lazy listener won't have a function anyway.
  • khtml/ecma/kjs_window.h: Prototype new method.
Location:
trunk/WebCore
Files:
9 edited

Legend:

Unmodified
Added
Removed
  • trunk/WebCore/ChangeLog-2005-08-23

    r5403 r5404  
     12003-11-06  Maciej Stachowiak  <mjs@apple.com>
     2
     3        Reviewed by Darin.
     4
     5        - fixed 3475366 - 4.5% of time spent making html event listeners on slow intel page.
     6
     7        6% speedup on intel page, 1% speedup on cvs-base PLT.
     8       
     9        * khtml/ecma/kjs_events.cpp:
     10        (JSEventListener::JSEventListener): Don't add self to hashtable if imp is null (which
     11        can now happen in the lazy listener case).
     12        (JSEventListener::~JSEventListener): Ditto on removing.
     13        (JSEventListener::listenerObj): Made this virtual.
     14        (JSLazyEventListener::JSLazyEventListener): New constructor.
     15        (JSLazyEventListener::handleEvent): call parseCode, then
     16        superclass if it appeared to succeed.
     17        (JSLazyEventListener::listenerObj): call parseCode, then superclass.
     18        (JSLazyEventListener::parseCode):
     19        (KJS::getNodeEventListener): Check for null listenerObjImp in case of
     20        lazy listener that failed to parse.
     21        * khtml/ecma/kjs_dom.cpp:
     22        (DOMNode::getListener): Ditto.
     23        * khtml/ecma/kjs_html.cpp:
     24        (Image::getValueProperty): Ditto.
     25        * khtml/ecma/kjs_events.h:
     26        (KJS::JSEventListener::listenerObjImp): call listenerObj() virtual
     27        method and get imp from the result.
     28        * khtml/ecma/kjs_proxy.cpp:
     29        (KJSProxyImpl::createHTMLEventHandler): Don't parse the code here, make a lazy
     30        listener.
     31        * khtml/ecma/kjs_window.cpp:
     32        (Window::getJSLazyEventListener): make a new JSLazyEventListener - no need
     33        to check the listeners hashtable cause a brand new lazy listener won't have
     34        a function anyway.
     35        * khtml/ecma/kjs_window.h: Prototype new method.
     36
    1372003-11-06  Maciej Stachowiak  <mjs@apple.com>
    238
  • trunk/WebCore/khtml/ecma/kjs_dom.cpp

    r5296 r5404  
    416416{
    417417    DOM::EventListener *listener = node.handle()->getHTMLEventListener(eventId);
    418     if (listener)
    419         return static_cast<JSEventListener*>(listener)->listenerObj();
     418    JSEventListener *jsListener = static_cast<JSEventListener*>(listener);
     419    if ( jsListener && jsListener->listenerObjImp() )
     420        return jsListener->listenerObj();
    420421    else
    421422        return Null();
  • trunk/WebCore/khtml/ecma/kjs_events.cpp

    r5402 r5404  
    4444    html = _html;
    4545    win = _win;
    46     static_cast<Window*>(win.imp())->jsEventListeners.insert(_listener.imp(), this);
     46    if (_listener.imp()) {
     47      static_cast<Window*>(win.imp())->jsEventListeners.insert(_listener.imp(), this);
     48    }
    4749}
    4850
    4951JSEventListener::~JSEventListener()
    5052{
    51     static_cast<Window*>(win.imp())->jsEventListeners.remove(listener.imp());
     53    if (listener.imp()) {
     54      static_cast<Window*>(win.imp())->jsEventListeners.remove(listener.imp());
     55    }
    5256    //fprintf(stderr,"JSEventListener::~JSEventListener this=%p listener=%p\n",this,listener.imp());
    5357}
     
    6064#endif
    6165  KHTMLPart *part = static_cast<Window*>(win.imp())->part();
    62   KJSProxy *proxy = 0L;
     66  KJSProxy *proxy = 0;
    6367  if (part)
    6468      proxy = KJSProxy::proxy( part );
     
    132136}
    133137
     138
     139Object JSEventListener::listenerObj() const
     140{
     141  return listener;
     142}
     143
     144JSLazyEventListener::JSLazyEventListener(QString _code, const Object &_win, bool _html)
     145  : JSEventListener(Object(), _win, _html),
     146    code(_code),
     147    parsed(false)
     148{
     149}
     150
     151void JSLazyEventListener::handleEvent(DOM::Event &evt, bool isWindowEvent)
     152{
     153  parseCode();
     154  if (!listener.isNull()) {
     155    JSEventListener::handleEvent(evt, isWindowEvent);
     156  }
     157}
     158
     159
     160Object JSLazyEventListener::listenerObj() const
     161{
     162  parseCode();
     163  return listener;
     164}
     165
     166void JSLazyEventListener::parseCode() const
     167{
     168  if (!parsed) {
     169    KHTMLPart *part = static_cast<Window*>(win.imp())->part();
     170    KJSProxy *proxy = 0L;
     171    if (part)
     172      proxy = KJSProxy::proxy( part );
     173
     174    if (proxy) {
     175      KJS::ScriptInterpreter *interpreter = static_cast<KJS::ScriptInterpreter *>(proxy->interpreter());
     176      ExecState *exec = interpreter->globalExec();
     177
     178      //KJS::Constructor constr(KJS::Global::current().get("Function").imp());
     179      KJS::Object constr = interpreter->builtinFunction();
     180      KJS::List args;
     181
     182      static KJS::String eventString("event");
     183
     184      args.append(eventString);
     185      args.append(KJS::String(code));
     186      listener = constr.construct(exec, args); // ### is globalExec ok ?
     187     
     188      if ( exec->hadException() ) {
     189        exec->clearException();
     190
     191        // failed to parse, so let's just make this listener a no-op
     192        listener = Object();
     193      }
     194    }
     195
     196    // no more need to keep the unparsed code around
     197    code = QString();
     198   
     199    if (!listener.isNull()) {
     200      static_cast<Window*>(win.imp())->jsEventListeners.insert(listener.imp(),
     201                                                               (KJS::JSEventListener *)(this));
     202    }
     203   
     204    parsed = true;
     205  }
     206}
     207
    134208Value KJS::getNodeEventListener(DOM::Node n, int eventId)
    135209{
    136210    DOM::EventListener *listener = n.handle()->getHTMLEventListener(eventId);
    137     if (listener)
    138         return static_cast<JSEventListener*>(listener)->listenerObj();
     211    JSEventListener *jsListener = static_cast<JSEventListener*>(listener);
     212    if ( jsListener && jsListener->listenerObjImp() )
     213        return jsListener->listenerObj();
    139214    else
    140215        return Null();
    141216}
     217
     218
    142219
    143220// -------------------------------------------------------------------------
  • trunk/WebCore/khtml/ecma/kjs_events.h

    r5402 r5404  
    3737    virtual void handleEvent(DOM::Event &evt, bool isWindowEvent);
    3838    virtual DOM::DOMString eventListenerType();
    39     Object listenerObj() const { return listener; }
    40     ObjectImp *listenerObjImp() const { return listener.imp(); }
     39    virtual Object listenerObj() const;
     40    ObjectImp *listenerObjImp() const { return listenerObj().imp(); }
    4141  protected:
    42     Object listener;
     42    mutable Object listener;
    4343    bool html;
    4444    Object win;
    4545  };
     46
     47  class JSLazyEventListener : public JSEventListener {
     48  public:
     49    JSLazyEventListener(QString _code, const Object &_win, bool _html = false);
     50    virtual void handleEvent(DOM::Event &evt, bool isWindowEvent);
     51    Object listenerObj() const;
     52  private:
     53    void parseCode() const;
     54   
     55    mutable QString code;
     56    mutable bool parsed;
     57  };
     58
    4659
    4760  Value getNodeEventListener(DOM::Node n, int eventId);
  • trunk/WebCore/khtml/ecma/kjs_html.cpp

    r5324 r5404  
    31863186    return Boolean(!img || img->status() >= khtml::CachedObject::Persistent);
    31873187  case OnLoad:
    3188     if (onLoadListener) {
     3188    if (onLoadListener && onLoadListener->listenerObjImp()) {
    31893189      return onLoadListener->listenerObj();
    31903190    } else {
  • trunk/WebCore/khtml/ecma/kjs_proxy.cpp

    r4666 r5404  
    165165
    166166  initScript();
    167   //KJS::Constructor constr(KJS::Global::current().get("Function").imp());
    168   KJS::Object constr = m_script->builtinFunction();
    169   KJS::List args;
    170   args.append(KJS::String("event"));
    171   args.append(KJS::String(code));
    172   Object handlerFunc = constr.construct(m_script->globalExec(), args); // ### is globalExec ok ?
    173 
    174   return KJS::Window::retrieveWindow(m_part)->getJSEventListener(handlerFunc,true);
     167
     168  return KJS::Window::retrieveWindow(m_part)->getJSLazyEventListener(code,true);
    175169}
    176170
  • trunk/WebCore/khtml/ecma/kjs_window.cpp

    r5394 r5404  
    10331033}
    10341034
     1035JSLazyEventListener *Window::getJSLazyEventListener(const QString& code, bool html)
     1036{
     1037  return new JSLazyEventListener(code, Object(this), html);
     1038}
     1039
    10351040void Window::clear( ExecState *exec )
    10361041{
  • trunk/WebCore/khtml/ecma/kjs_window.h

    r5394 r5404  
    4242  class FrameArray;
    4343  class JSEventListener;
     44  class JSLazyEventListener;
    4445
    4546  class Screen : public ObjectImp {
     
    101102    Location *location() const;
    102103    JSEventListener *getJSEventListener(const Value &val, bool html = false);
     104    JSLazyEventListener *getJSLazyEventListener(const QString &code, bool html = false);
    103105    void clear( ExecState *exec );
    104106    virtual UString toString(ExecState *exec) const;
  • trunk/WebCore/khtml/xml/dom_docimpl.cpp

    r5391 r5404  
    23972397QString DocumentImpl::completeURL(const QString &URL)
    23982398{
    2399     return KURL(baseURL(), URL, m_decoder ? m_decoder->codec() : 0).url();
     2399*    return KURL(baseURL(), URL, m_decoder ? m_decoder->codec() : 0).url();
    24002400}
    24012401
Note: See TracChangeset for help on using the changeset viewer.