/**
* Copyright 2012 Tobias Gierke <tobias.gierke@code-sourcery.de>
*
* Licensed under the Apache License, Version 2.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.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
package de.codesourcery.jasm16.ast;
import java.io.IOException;
import de.codesourcery.jasm16.Address;
import de.codesourcery.jasm16.compiler.ICompilationContext;
import de.codesourcery.jasm16.compiler.io.IObjectCodeWriter;
import de.codesourcery.jasm16.exceptions.ParseException;
import de.codesourcery.jasm16.lexer.TokenType;
import de.codesourcery.jasm16.parser.IParseContext;
/**
* '.org' AST node.
*
* @author tobias.gierke@code-sourcery.de
*/
public class OriginNode extends ObjectCodeOutputNode implements IPreprocessorDirective
{
@Override
protected ASTNode copySingleNode()
{
final OriginNode result = new OriginNode();
result.setAddress( getAddress() );
return result;
}
@Override
public boolean supportsChildNodes()
{
return true;
}
@Override
protected ASTNode parseInternal(IParseContext context) throws ParseException
{
mergeWithAllTokensTextRegion( context.read( TokenType.ORIGIN ) );
mergeWithAllTokensTextRegion( context.parseWhitespace() );
final NumberNode number;
try {
context.mark();
number = (NumberNode) addChild( new NumberNode().parseInternal( context ) , context );
} catch(Exception e) {
addCompilationErrorAndAdvanceParser( e , context );
return this;
} finally {
context.clearMark();
}
try {
setAddress( Address.wordAddress( number.getValue() ) );
} catch(IllegalArgumentException e) {
context.addCompilationError( "Address value is out-of-range" , number );
}
return this;
}
@Override
public void symbolsResolved(ICompilationContext context)
{
// nothing to do here
}
@Override
public int getSizeInBytes(long thisNodesObjectCodeOffsetInBytes)
{
if ( getAddress().getValue() < thisNodesObjectCodeOffsetInBytes ) {
return UNKNOWN_SIZE;
} else if ( getAddress().getValue() == thisNodesObjectCodeOffsetInBytes ) {
return 0;
}
return (int) (getAddress().getValue() - thisNodesObjectCodeOffsetInBytes);
}
@Override
public void writeObjectCode(IObjectCodeWriter writer, ICompilationContext compContext) throws IOException, ParseException
{
if ( getAddress() == null ) {
throw new RuntimeException("Internal error, .origin node has no address ?");
}
writer.advanceToWriteOffset( getAddress() );
}
}