/***** BEGIN LICENSE BLOCK ***** * Version: CPL 1.0/GPL 2.0/LGPL 2.1 * * The contents of this file are subject to the Common Public * License Version 1.0 (the "License"); you may not use this file * except in compliance with the License. You may obtain a copy of * the License at http://www.eclipse.org/legal/cpl-v10.html * * Software distributed under the License is distributed on an "AS * IS" basis, WITHOUT WARRANTY OF ANY KIND, either express or * implied. See the License for the specific language governing * rights and limitations under the License. * * Copyright (C) 2002-2004 Anders Bengtsson <ndrsbngtssn@yahoo.se> * Copyright (C) 2002-2004 Jan Arne Petersen <jpetersen@uni-bonn.de> * Copyright (C) 2004 Stefan Matthias Aust <sma@3plus4.de> * * Alternatively, the contents of this file may be used under the terms of * either of the GNU General Public License Version 2 or later (the "GPL"), * or the GNU Lesser General Public License Version 2.1 or later (the "LGPL"), * in which case the provisions of the GPL or the LGPL are applicable instead * of those above. If you wish to allow use of your version of this file only * under the terms of either the GPL or the LGPL, and not to allow others to * use your version of this file under the terms of the CPL, indicate your * decision by deleting the provisions above and replace them with the notice * and other provisions required by the GPL or the LGPL. If you do not delete * the provisions above, a recipient may use your version of this file under * the terms of any one of the CPL, the GPL or the LGPL. ***** END LICENSE BLOCK *****/ package org.jruby.runtime; import java.io.Serializable; import java.util.HashMap; import java.util.Map; import org.jruby.ast.AttrAssignNode; import org.jruby.ast.CallNode; import org.jruby.ast.Node; import org.jruby.ast.types.IArityNode; /** * The arity of a method is the number of arguments it takes. */ public final class Arity implements Serializable { private static final long serialVersionUID = 1L; private static final Map arities = new HashMap(); private final int value; public final static Arity NO_ARGUMENTS = newArity(0); public final static Arity ONE_ARGUMENT = newArity(1); public final static Arity TWO_ARGUMENTS = newArity(2); public final static Arity THREE_ARGUMENTS = newArity(3); public final static Arity OPTIONAL = newArity(-1); public final static Arity ONE_REQUIRED = newArity(-2); public final static Arity TWO_REQUIRED = newArity(-3); public final static Arity THREE_REQUIRED = newArity(-3); private Arity(int value) { this.value = value; } public static Arity createArity(int value) { switch (value) { case -4: return THREE_REQUIRED; case -3: return TWO_REQUIRED; case -2: return ONE_REQUIRED; case -1: return OPTIONAL; case 0: return NO_ARGUMENTS; case 1: return ONE_ARGUMENT; case 2: return TWO_ARGUMENTS; case 3: return THREE_ARGUMENTS; } return newArity(value); } private static Arity newArity(int value) { Integer integerValue = new Integer(value); Arity result; synchronized (arities) { result = (Arity) arities.get(integerValue); if (result == null) { result = new Arity(value); arities.put(integerValue, result); } } return result; } public static Arity fixed(int arity) { // assert arity >= 0; return createArity(arity); } public static Arity optional() { return OPTIONAL; } public static Arity required(int minimum) { // assert minimum >= 0; return createArity(-(1 + minimum)); } public static Arity noArguments() { return NO_ARGUMENTS; } public static Arity singleArgument() { return ONE_ARGUMENT; } public static Arity twoArguments() { return TWO_ARGUMENTS; } public static Arity procArityOf(Node node) { if (node instanceof AttrAssignNode && node != null) { node = ((AttrAssignNode) node).getArgsNode(); } if (node == null) { return Arity.optional(); } else if (node instanceof IArityNode) { return ((IArityNode) node).getArity(); } else if (node instanceof CallNode) { return Arity.singleArgument(); } throw new Error("unexpected type " + node.getClass() + " at " + node.getPosition()); } public int getValue() { return value; } // public void checkArity(Ruby runtime, IRubyObject[] args) { // if (isFixed()) { // if (args.length != required()) { // throw runtime.newArgumentError("wrong number of arguments(" + args.length + " for " + required() + ")"); // } // } else { // if (args.length < required()) { // throw runtime.newArgumentError("wrong number of arguments(" + args.length + " for " + required() + ")"); // } // } // } public boolean isFixed() { return value >= 0; } public int required() { if (value < 0) { return -(1 + value); } return value; } public boolean equals(Object other) { return this == other; } public int hashCode() { return value; } public String toString() { if(isFixed()) { return "Fixed" + required(); } else { return "Opt"; } } // Some helper functions: // public static int checkArgumentCount(Ruby runtime, IRubyObject[] args, int min, int max) { // if (args.length < min) { // throw runtime.newArgumentError("wrong number of arguments (" + args.length + " for " + min + ")"); // } // if (max > -1 && args.length > max) { // throw runtime.newArgumentError("wrong number of arguments (" + args.length + " for " + max + ")"); // } // return args.length; // } // // /** // * @see org.jruby.runtime.builtin.IRubyObject#scanArgs() // */ // public static IRubyObject[] scanArgs(Ruby runtime, IRubyObject[] args, int required, int optional) { // int total = required+optional; // int real = checkArgumentCount(runtime, args,required,total); // IRubyObject[] narr = new IRubyObject[total]; // System.arraycopy(args,0,narr,0,real); // for(int i=real; i<total; i++) { // narr[i] = runtime.getNil(); // } // return narr; // } }