/* GNU Objective C Runtime method related functions.
   Copyright (C) 2010-2020 Free Software Foundation, Inc.
   Contributed by Nicola Pero
This file is part of GCC.
GCC 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, or (at your option) any later version.
GCC 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.
Under Section 7 of GPL version 3, you are granted additional
permissions described in the GCC Runtime Library Exception, version
3.1, as published by the Free Software Foundation.
You should have received a copy of the GNU General Public License and
a copy of the GCC Runtime Library Exception along with this program;
see the files COPYING3 and COPYING.RUNTIME respectively.  If not, see
.  */
#include "objc-private/common.h"
#include "objc/runtime.h"
#include "objc-private/module-abi-8.h" /* For runtime structures.   */
#include "objc/thr.h"
#include "objc-private/runtime.h"      /* For __objc_runtime_mutex.  */
#include                     /* For malloc.  */
SEL
method_getName (struct objc_method * method)
{
  if (method == NULL)
    return NULL;
  return method->method_name;
}
const char *
method_getTypeEncoding (struct objc_method * method)
{
  if (method == NULL)
    return NULL;
  return method->method_types;
}
IMP
method_getImplementation (struct objc_method * method)
{
  if (method == NULL)
    return NULL;
  return method->method_imp;
}
struct objc_method_description *
method_getDescription (struct objc_method * method)
{
  /* Note that the following returns NULL if method is NULL, which is
     fine.  */
  return (struct objc_method_description *)method;
}
struct objc_method **
class_copyMethodList (Class class_, unsigned int *numberOfReturnedMethods)
{
  unsigned int count = 0;
  struct objc_method **returnValue = NULL;
  struct objc_method_list* method_list;
  if (class_ == Nil)
    {
      if (numberOfReturnedMethods)
	*numberOfReturnedMethods = 0;
      return NULL;
    }
  /* Lock the runtime mutex because the class methods may be
     concurrently modified.  */
  objc_mutex_lock (__objc_runtime_mutex);
  /* Count how many methods we have.  */
  method_list = class_->methods;
  while (method_list)
    {
      count = count + method_list->method_count;
      method_list = method_list->method_next;
    }
  if (count != 0)
    {
      unsigned int i = 0;
      
      /* Allocate enough memory to hold them.  */
      returnValue 
	= (struct objc_method **)(malloc (sizeof (struct objc_method *) 
					  * (count + 1)));
      
      /* Copy the methods.  */
      method_list = class_->methods;
      
      while (method_list)
	{
	  int j;
	  for (j = 0; j < method_list->method_count; j++)
	    {
	      returnValue[i] = &(method_list->method_list[j]);
	      i++;
	    }
	  method_list = method_list->method_next;
	}
      
      returnValue[i] = NULL;
    }
  objc_mutex_unlock (__objc_runtime_mutex);
  if (numberOfReturnedMethods)
    *numberOfReturnedMethods = count;
  return returnValue;
}
IMP
method_setImplementation (struct objc_method * method, IMP implementation)
{
  IMP old_implementation;
  if (method == NULL  ||  implementation == NULL)
    return NULL;
  /* We lock the runtime mutex so that concurrent calls to change the
     same method won't conflict with each other.  */
  objc_mutex_lock (__objc_runtime_mutex);
  old_implementation = method->method_imp;
  method->method_imp = implementation;
  /* That was easy :-).  But now we need to find all classes that use
     this method, and update the IMP in the dispatch tables.  */
  __objc_update_classes_with_methods (method, NULL);
  objc_mutex_unlock (__objc_runtime_mutex);
  return old_implementation;
}
void
method_exchangeImplementations (struct objc_method * method_a, struct objc_method * method_b)
{
  IMP old_implementation_a;
  IMP old_implementation_b;
  if (method_a == NULL  ||  method_b == NULL)
    return;
  /* We lock the runtime mutex so that concurrent calls to exchange
     similar methods won't conflict with each other.  Each of them
     should be atomic.  */
  objc_mutex_lock (__objc_runtime_mutex);
  old_implementation_a = method_a->method_imp;
  old_implementation_b = method_b->method_imp;
  method_a->method_imp = old_implementation_b;
  method_b->method_imp = old_implementation_a;
  /* That was easy :-).  But now we need to find all classes that use
     these methods, and update the IMP in the dispatch tables.  */
  __objc_update_classes_with_methods (method_a, method_b);
  objc_mutex_unlock (__objc_runtime_mutex);
}