tracesrv/tracecompiler/src/com.nokia.tracecompiler/src/com/nokia/tracecompiler/engine/rules/AutoAddReturnParameterRule.java
author hgs
Fri, 08 Oct 2010 14:56:39 +0300
changeset 56 aa2539c91954
permissions -rw-r--r--
201041
Ignore whitespace changes - Everywhere: Within whitespace: At end of lines:
56
hgs
parents:
diff changeset
     1
/*
hgs
parents:
diff changeset
     2
* Copyright (c) 2008 Nokia Corporation and/or its subsidiary(-ies). 
hgs
parents:
diff changeset
     3
* All rights reserved.
hgs
parents:
diff changeset
     4
* This component and the accompanying materials are made available
hgs
parents:
diff changeset
     5
* under the terms of "Eclipse Public License v1.0"
hgs
parents:
diff changeset
     6
* which accompanies this distribution, and is available
hgs
parents:
diff changeset
     7
* at the URL "http://www.eclipse.org/legal/epl-v10.html".
hgs
parents:
diff changeset
     8
*
hgs
parents:
diff changeset
     9
* Initial Contributors:
hgs
parents:
diff changeset
    10
* Nokia Corporation - initial contribution.
hgs
parents:
diff changeset
    11
*
hgs
parents:
diff changeset
    12
* Contributors:
hgs
parents:
diff changeset
    13
*
hgs
parents:
diff changeset
    14
* Description:
hgs
parents:
diff changeset
    15
*
hgs
parents:
diff changeset
    16
* Trace rule for automatically adding function return value to exit trace
hgs
parents:
diff changeset
    17
*
hgs
parents:
diff changeset
    18
*/
hgs
parents:
diff changeset
    19
package com.nokia.tracecompiler.engine.rules;
hgs
parents:
diff changeset
    20
hgs
parents:
diff changeset
    21
import java.util.ArrayList;
hgs
parents:
diff changeset
    22
hgs
parents:
diff changeset
    23
import com.nokia.tracecompiler.engine.TraceCompilerEngineErrorMessages;
hgs
parents:
diff changeset
    24
import com.nokia.tracecompiler.engine.TraceCompilerEngineGlobals;
hgs
parents:
diff changeset
    25
import com.nokia.tracecompiler.engine.TraceCompilerEngineErrorCodes.TraceCompilerErrorCode;
hgs
parents:
diff changeset
    26
import com.nokia.tracecompiler.engine.utils.TraceMultiplierRule;
hgs
parents:
diff changeset
    27
import com.nokia.tracecompiler.model.Trace;
hgs
parents:
diff changeset
    28
import com.nokia.tracecompiler.model.TraceCompilerException;
hgs
parents:
diff changeset
    29
import com.nokia.tracecompiler.model.TraceConstantTable;
hgs
parents:
diff changeset
    30
import com.nokia.tracecompiler.model.TraceObjectRuleCreateObject;
hgs
parents:
diff changeset
    31
import com.nokia.tracecompiler.model.TraceObjectRuleRemoveOnCreate;
hgs
parents:
diff changeset
    32
import com.nokia.tracecompiler.model.TraceParameter;
hgs
parents:
diff changeset
    33
import com.nokia.tracecompiler.source.SourceContext;
hgs
parents:
diff changeset
    34
import com.nokia.tracecompiler.source.SourceParserException;
hgs
parents:
diff changeset
    35
import com.nokia.tracecompiler.source.SourceReturn;
hgs
parents:
diff changeset
    36
import com.nokia.tracecompiler.source.SourceUtils;
hgs
parents:
diff changeset
    37
import com.nokia.tracecompiler.source.TypeMapping;
hgs
parents:
diff changeset
    38
hgs
parents:
diff changeset
    39
/**
hgs
parents:
diff changeset
    40
 * Trace rule for automatically adding function return value to exit trace
hgs
parents:
diff changeset
    41
 * 
hgs
parents:
diff changeset
    42
 */
hgs
parents:
diff changeset
    43
public final class AutoAddReturnParameterRule extends RuleBase implements
hgs
parents:
diff changeset
    44
		TraceObjectRuleCreateObject, TraceObjectRuleRemoveOnCreate {
hgs
parents:
diff changeset
    45
hgs
parents:
diff changeset
    46
	/**
hgs
parents:
diff changeset
    47
	 * Parameter name
hgs
parents:
diff changeset
    48
	 */
hgs
parents:
diff changeset
    49
	static final String PARAMETER_NAME = "retval"; //$NON-NLS-1$
hgs
parents:
diff changeset
    50
hgs
parents:
diff changeset
    51
	/**
hgs
parents:
diff changeset
    52
	 * The number of handled return statements
hgs
parents:
diff changeset
    53
	 */
hgs
parents:
diff changeset
    54
	private static int numberOfHandledReturnStatements = 0;
hgs
parents:
diff changeset
    55
hgs
parents:
diff changeset
    56
	/*
hgs
parents:
diff changeset
    57
	 * (non-Javadoc)
hgs
parents:
diff changeset
    58
	 * 
hgs
parents:
diff changeset
    59
	 * @see com.nokia.tracecompiler.model.TraceObjectRuleCreateObject#
hgs
parents:
diff changeset
    60
	 *      createObject(com.nokia.tracecompiler.model.TraceObject)
hgs
parents:
diff changeset
    61
	 */
hgs
parents:
diff changeset
    62
	public void createObject() throws TraceCompilerException {
hgs
parents:
diff changeset
    63
		// If owner has a multiplier (entry trace), the trace is not added to it
hgs
parents:
diff changeset
    64
		Trace owner = (Trace) getOwner();
hgs
parents:
diff changeset
    65
		if (owner.getExtension(TraceMultiplierRule.class) == null) {
hgs
parents:
diff changeset
    66
			SourceContext context = TraceCompilerEngineGlobals
hgs
parents:
diff changeset
    67
					.getSourceContextManager().getContext();
hgs
parents:
diff changeset
    68
			if (!context.isVoid()) {
hgs
parents:
diff changeset
    69
				TraceConstantTable table = RuleUtils.findConstantTableByType(
hgs
parents:
diff changeset
    70
						owner.getModel(), context);
hgs
parents:
diff changeset
    71
				if (table != null) {
hgs
parents:
diff changeset
    72
					createParameter(owner, table);
hgs
parents:
diff changeset
    73
				} else {
hgs
parents:
diff changeset
    74
					createParameter(owner, context);
hgs
parents:
diff changeset
    75
				}
hgs
parents:
diff changeset
    76
			}
hgs
parents:
diff changeset
    77
		}
hgs
parents:
diff changeset
    78
	}
hgs
parents:
diff changeset
    79
hgs
parents:
diff changeset
    80
	/*
hgs
parents:
diff changeset
    81
	 * (non-Javadoc)
hgs
parents:
diff changeset
    82
	 * 
hgs
parents:
diff changeset
    83
	 * @see com.nokia.tracecompiler.model.TraceObjectRuleRemoveOnCreate#canBeRemoved()
hgs
parents:
diff changeset
    84
	 */
hgs
parents:
diff changeset
    85
	public boolean canBeRemoved() {
hgs
parents:
diff changeset
    86
		// If the owner has a multiplier, this needs to be moved to it using the
hgs
parents:
diff changeset
    87
		// CopyAndRemoveExtensionRule
hgs
parents:
diff changeset
    88
		return getOwner().getExtension(TraceMultiplierRule.class) == null;
hgs
parents:
diff changeset
    89
	}
hgs
parents:
diff changeset
    90
hgs
parents:
diff changeset
    91
	/**
hgs
parents:
diff changeset
    92
	 * Creates a return parameter that was not associated with a constant table
hgs
parents:
diff changeset
    93
	 * 
hgs
parents:
diff changeset
    94
	 * @param owner
hgs
parents:
diff changeset
    95
	 *            the owner for the parameter
hgs
parents:
diff changeset
    96
	 * @param context
hgs
parents:
diff changeset
    97
	 *            the context specifying the parameter type 
hgs
parents:
diff changeset
    98
	 */
hgs
parents:
diff changeset
    99
	private void createParameter(Trace owner, SourceContext context) {
hgs
parents:
diff changeset
   100
		TypeMapping type = SourceUtils.mapSymbianTypeToParameterType(context);
hgs
parents:
diff changeset
   101
		String return_value_name = PARAMETER_NAME;
hgs
parents:
diff changeset
   102
		ArrayList<SourceReturn> returnList = new ArrayList<SourceReturn>();
hgs
parents:
diff changeset
   103
hgs
parents:
diff changeset
   104
		// Find out return value name
hgs
parents:
diff changeset
   105
		try {
hgs
parents:
diff changeset
   106
			context.parseReturnValues(returnList);
hgs
parents:
diff changeset
   107
			if (numberOfHandledReturnStatements <= (returnList.size() - 1)) {
hgs
parents:
diff changeset
   108
				return_value_name = returnList.get(
hgs
parents:
diff changeset
   109
						numberOfHandledReturnStatements).getReturnStatement();
hgs
parents:
diff changeset
   110
			}
hgs
parents:
diff changeset
   111
hgs
parents:
diff changeset
   112
			numberOfHandledReturnStatements++;
hgs
parents:
diff changeset
   113
		} catch (SourceParserException e) {
hgs
parents:
diff changeset
   114
			String msg = Messages
hgs
parents:
diff changeset
   115
					.getString("RuleUtils.FailedToParseReturnValues"); //$NON-NLS-1$
hgs
parents:
diff changeset
   116
			String cname = context.getClassName();
hgs
parents:
diff changeset
   117
			String source;
hgs
parents:
diff changeset
   118
			if (cname != null) {
hgs
parents:
diff changeset
   119
				source = cname + "::" + context.getFunctionName(); //$NON-NLS-1$;
hgs
parents:
diff changeset
   120
			} else {
hgs
parents:
diff changeset
   121
				source = context.getFunctionName();
hgs
parents:
diff changeset
   122
			}
hgs
parents:
diff changeset
   123
			TraceCompilerEngineGlobals.getEvents().postErrorMessage(msg, source, true);
hgs
parents:
diff changeset
   124
		}
hgs
parents:
diff changeset
   125
hgs
parents:
diff changeset
   126
		try {
hgs
parents:
diff changeset
   127
			if (!type.needsCasting) {
hgs
parents:
diff changeset
   128
				// If an extension header is generated, the parameter needs to
hgs
parents:
diff changeset
   129
				// be cast, since the return statement may contain anything.
hgs
parents:
diff changeset
   130
				type.needsCasting = type.type != TraceParameter.HEX32
hgs
parents:
diff changeset
   131
						&& type.type != TraceParameter.SDEC32
hgs
parents:
diff changeset
   132
						&& type.type != TraceParameter.UDEC32;
hgs
parents:
diff changeset
   133
			}
hgs
parents:
diff changeset
   134
			RuleUtils.createParameterFromType(owner, return_value_name, type);
hgs
parents:
diff changeset
   135
hgs
parents:
diff changeset
   136
		} catch (TraceCompilerException e) {
hgs
parents:
diff changeset
   137
hgs
parents:
diff changeset
   138
			if (e.getErrorCode() == TraceCompilerErrorCode.INVALID_PARAMETER_NAME) {
hgs
parents:
diff changeset
   139
				String msg = TraceCompilerEngineErrorMessages
hgs
parents:
diff changeset
   140
						.getErrorMessage(
hgs
parents:
diff changeset
   141
								TraceCompilerErrorCode.INVALID_PARAMETER_NAME_IN_RETURN_VALUE,
hgs
parents:
diff changeset
   142
								null);
hgs
parents:
diff changeset
   143
hgs
parents:
diff changeset
   144
				TraceCompilerEngineGlobals.getEvents().postWarningMessage(msg, owner);
hgs
parents:
diff changeset
   145
			} else {
hgs
parents:
diff changeset
   146
				TraceCompilerException exception = new TraceCompilerException(e.getErrorCode(), e
hgs
parents:
diff changeset
   147
						.getErrorParameters(), null);
hgs
parents:
diff changeset
   148
				TraceCompilerEngineGlobals.getEvents().postError(
hgs
parents:
diff changeset
   149
						exception);
hgs
parents:
diff changeset
   150
			}
hgs
parents:
diff changeset
   151
		}
hgs
parents:
diff changeset
   152
	}
hgs
parents:
diff changeset
   153
hgs
parents:
diff changeset
   154
	/**
hgs
parents:
diff changeset
   155
	 * Creates a return parameter that was associated with a constant table
hgs
parents:
diff changeset
   156
	 * 
hgs
parents:
diff changeset
   157
	 * @param owner
hgs
parents:
diff changeset
   158
	 *            the owner for the parameter
hgs
parents:
diff changeset
   159
	 * @param table
hgs
parents:
diff changeset
   160
	 *            the constant table the parameter was associated to
hgs
parents:
diff changeset
   161
	 */
hgs
parents:
diff changeset
   162
	private void createParameter(Trace owner, TraceConstantTable table) {
hgs
parents:
diff changeset
   163
		try {
hgs
parents:
diff changeset
   164
			RuleUtils.createParameterFromConstantTable(owner, PARAMETER_NAME,
hgs
parents:
diff changeset
   165
					table);
hgs
parents:
diff changeset
   166
		} catch (TraceCompilerException e) {
hgs
parents:
diff changeset
   167
			TraceCompilerException exception = new TraceCompilerException(e.getErrorCode(), e
hgs
parents:
diff changeset
   168
					.getErrorParameters(), null);
hgs
parents:
diff changeset
   169
			TraceCompilerEngineGlobals.getEvents().postError(
hgs
parents:
diff changeset
   170
					exception);
hgs
parents:
diff changeset
   171
		}
hgs
parents:
diff changeset
   172
	}
hgs
parents:
diff changeset
   173
hgs
parents:
diff changeset
   174
	/**
hgs
parents:
diff changeset
   175
	 * Reset static variables
hgs
parents:
diff changeset
   176
	 */
hgs
parents:
diff changeset
   177
	static void resetNumberOfHandledReturnStatements() {
hgs
parents:
diff changeset
   178
		numberOfHandledReturnStatements = 0;
hgs
parents:
diff changeset
   179
	}
hgs
parents:
diff changeset
   180
hgs
parents:
diff changeset
   181
	/**
hgs
parents:
diff changeset
   182
	 * Increase number of handled return statements
hgs
parents:
diff changeset
   183
	 */
hgs
parents:
diff changeset
   184
	static void increaseNumberOfHandledReturnStatements() {
hgs
parents:
diff changeset
   185
		numberOfHandledReturnStatements++;
hgs
parents:
diff changeset
   186
	}
hgs
parents:
diff changeset
   187
hgs
parents:
diff changeset
   188
}