view libinterp/corefcn/fcn-info.h @ 31231:a026fb2be108

sparse-xpow.cc: Return empty matrix for empty input (bug #63080)
author Arun Giridhar <arungiridhar@gmail.com>
date Mon, 19 Sep 2022 07:05:31 -0400
parents 796f54d4ddbf
children e88a07dec498
line wrap: on
line source

////////////////////////////////////////////////////////////////////////
//
// Copyright (C) 1993-2022 The Octave Project Developers
//
// See the file COPYRIGHT.md in the top-level directory of this
// distribution or <https://octave.org/copyright/>.
//
// This file is part of Octave.
//
// Octave is free software: you can redistribute it and/or modify it
// under the terms of the GNU General Public License as published by
// the Free Software Foundation, either version 3 of the License, or
// (at your option) any later version.
//
// Octave is distributed in the hope that it will be useful, but
// WITHOUT ANY WARRANTY; without even the implied warranty of
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
// GNU General Public License for more details.
//
// You should have received a copy of the GNU General Public License
// along with Octave; see the file COPYING.  If not, see
// <https://www.gnu.org/licenses/>.
//
////////////////////////////////////////////////////////////////////////

#if ! defined (octave_fcn_info_h)
#define octave_fcn_info_h 1

#include "octave-config.h"

#include <list>
#include <map>
#include <memory>
#include <string>

#include "ov.h"
#include "ovl.h"
#include "symscope.h"

OCTAVE_NAMESPACE_BEGIN

  class fcn_info
  {
  public:

    typedef std::map<std::string, octave_value>::const_iterator
    str_val_const_iterator;
    typedef std::map<std::string, octave_value>::iterator str_val_iterator;

  private:

    class fcn_info_rep
    {
    public:

      fcn_info_rep (const std::string& nm)
        : name (nm), package_name (), local_functions (),
          private_functions (), class_constructors (), class_methods (),
          cmdline_function (), autoload_function (), function_on_path (),
          built_in_function ()
      {
        std::size_t pos = name.rfind ('.');

        if (pos != std::string::npos)
          {
            package_name = name.substr (0, pos);
            name = name.substr (pos+1);
          }
      }

      // No copying!

      fcn_info_rep (const fcn_info_rep&) = delete;

      fcn_info_rep& operator = (const fcn_info_rep&) = delete;

      ~fcn_info_rep (void) = default;

      octave_value install_local_function (const std::string& file_name);

      octave_value load_private_function (const std::string& dir_name);

      octave_value load_class_constructor (void);

      octave_value load_class_method (const std::string& dispatch_type);

      octave_value find (const symbol_scope& search_scope,
                         const octave_value_list& args);

      octave_value builtin_find (const symbol_scope& search_scope);

      octave_value find_scoped_function (const symbol_scope& search_scope);

      octave_value find_private_function (const std::string& dir_name);

      octave_value find_method (const std::string& dispatch_type);

      octave_value find_method (const octave_value_list& args);

      octave_value find_autoload (void);

      octave_value find_package (void);

      octave_value find_user_function (void);

      bool is_user_function_defined (void) const
      {
        return function_on_path.is_defined ();
      }

      octave_value find_function (const symbol_scope& search_scope,
                                  const octave_value_list& args)
      {
        return find (search_scope, args);
      }

      void install_cmdline_function (const octave_value& f)
      {
        cmdline_function = f;
      }

      void install_local_function (const octave_value& f,
                                   const std::string& file_name)
      {
        local_functions[file_name] = f;
      }

      void install_user_function (const octave_value& f)
      {
        function_on_path = f;
      }

      void install_built_in_function (const octave_value& f)
      {
        built_in_function = f;
      }

      void install_built_in_dispatch (const std::string& klass);

      template <typename T>
      void
      clear_map (std::map<T, octave_value>& map, bool force = false)
      {
        auto p = map.begin ();

        while (p != map.end ())
          {
            if (force || ! p->second.islocked ())
              map.erase (p++);
            else
              p++;
          }
      }

      void clear_autoload_function (bool force = false)
      {
        if (force || ! autoload_function.islocked ())
          autoload_function = octave_value ();
      }

      // We also clear command line functions here, as these are both
      // "user defined"
      void clear_user_function (bool force = false)
      {
        clear_autoload_function (force);

        if (force || ! function_on_path.islocked ())
          function_on_path = octave_value ();

        if (force || ! cmdline_function.islocked ())
          cmdline_function = octave_value ();
      }

      void clear_mex_function (void)
      {
        if (function_on_path.is_mex_function ())
          clear_user_function ();
      }

      void clear_package (void)
      {
        package = octave_value ();
      }

      void clear (bool force = false)
      {
        clear_map (local_functions, force);
        clear_map (private_functions, force);
        clear_map (class_constructors, force);
        clear_map (class_methods, force);

        clear_autoload_function (force);
        clear_user_function (force);
        clear_package ();
      }

      octave_value dump (void) const;

      std::string full_name (void) const
      {
        if (package_name.empty ())
          return name;
        else
          return package_name + '.' + name;
      }

      std::string name;

      std::string package_name;

      // File name to function object.
      std::map<std::string, octave_value> local_functions;

      // Directory name to function object.
      std::map<std::string, octave_value> private_functions;

      // Class name to function object.
      std::map<std::string, octave_value> class_constructors;

      // Dispatch type to function object.
      std::map<std::string, octave_value> class_methods;

      octave_value cmdline_function;

      octave_value autoload_function;

      octave_value function_on_path;

      octave_value package;

      octave_value built_in_function;

    private:

      octave_value xfind (const symbol_scope& search_scope,
                          const octave_value_list& args);

      octave_value x_builtin_find (const symbol_scope& search_scope);
    };

  public:

    fcn_info (const std::string& nm = "")
      : m_rep (new fcn_info_rep (nm)) { }

    fcn_info (const fcn_info&) = default;

    fcn_info& operator = (const fcn_info&) = default;

    ~fcn_info (void) = default;

    octave_value find (const symbol_scope& search_scope,
                       const octave_value_list& args = octave_value_list ())
    {
      return m_rep->find (search_scope, args);
    }

    octave_value
    builtin_find (const symbol_scope& search_scope)
    {
      return m_rep->builtin_find (search_scope);
    }

    octave_value find_scoped_function (const symbol_scope& search_scope) const
    {
      return m_rep->find_scoped_function (search_scope);
    }

    octave_value find_private_function (const std::string& dir_name) const
    {
      return m_rep->find_private_function (dir_name);
    }

    octave_value find_method (const std::string& dispatch_type) const
    {
      return m_rep->find_method (dispatch_type);
    }

    octave_value find_built_in_function (void) const
    {
      return m_rep->built_in_function;
    }

    octave_value find_cmdline_function (void) const
    {
      return m_rep->cmdline_function;
    }

    octave_value find_autoload (void)
    {
      return m_rep->find_autoload ();
    }

    // FIXME: find_function_on_path might be a better name?
    octave_value find_user_function (void)
    {
      return m_rep->find_user_function ();
    }

    bool is_user_function_defined (void) const
    {
      return m_rep->is_user_function_defined ();
    }

    octave_value
    find_function (const symbol_scope& search_scope,
                   const octave_value_list& args = octave_value_list ())
    {
      return m_rep->find_function (search_scope, args);
    }

    void install_cmdline_function (const octave_value& f)
    {
      m_rep->install_cmdline_function (f);
    }

    void install_local_function (const octave_value& f,
                                 const std::string& file_name)
    {
      m_rep->install_local_function (f, file_name);
    }

    void install_user_function (const octave_value& f)
    {
      m_rep->install_user_function (f);
    }

    void install_built_in_function (const octave_value& f)
    {
      m_rep->install_built_in_function (f);
    }

    void install_built_in_dispatch (const std::string& klass)
    {
      m_rep->install_built_in_dispatch (klass);
    }

    void clear (bool force = false) { m_rep->clear (force); }

    void clear_user_function (bool force = false)
    {
      m_rep->clear_user_function (force);
    }

    void clear_autoload_function (bool force = false)
    {
      m_rep->clear_autoload_function (force);
    }

    void clear_mex_function (void) { m_rep->clear_mex_function (); }

    octave_value dump (void) const { return m_rep->dump (); }

  private:

    std::shared_ptr<fcn_info_rep> m_rep;
  };

  extern OCTINTERP_API std::string
  get_dispatch_type (const octave_value_list& args);

  extern OCTINTERP_API std::string
  get_dispatch_type (const octave_value_list& args,
                     builtin_type_t& builtin_type);

  extern octave_value
  dump_function_map (const std::map<std::string, octave_value>& fcn_map);

OCTAVE_NAMESPACE_END

#endif