| Home » Eclipse Projects » Eclipse Titan » Eclipse Titan and the dilemma of PER-encoding(ASN.1 PER-codec for Titan )
 Goto Forum:| 
| Eclipse Titan and the dilemma of PER-encoding [message #1708432] | Wed, 16 September 2015 09:02  |  | 
| Eclipse User  |  |  |  |  | I think this subject needs a bit of an introduction, so let me go through some ASN.1 basics first:
 
 ASN.1 is a notation to describe structures , typically used as messages in communication systems.  ASN.1 offers an abstract syntax , that is,  a detailed structural description, with no reference to technology or implementation details;  and  a transfer syntax, a description of how the abstract structures are sent over the wire;  this is more commonly known as encoding or serialization.
 
 ASN.1 standardizes several of such encodings: XER (XML Encoding Rules) for XML encoding , BER(Basic Encoding Rules) for a binary, byte-oriented encoding,  and PER (Packed Encoding Rules), for a bit stream-oriented, compact encoding. The latter two will be of interest for us here.
 
 BER is a simple TLV(Type-Length-Value) encoding, meaning that types are assigned predefined codes,  and structures are encoded as a succession consisting of a type code, a field  signifying the  length of the data encoded, and the data itself in a binary form. BER-encoded messages can be easily
 decoded even without having access to the ASN.1 (minus the names  assigned to structures of course, which are not transmitted over the wire).
 
 PER is another piece of cake: it uses the minimum amount of bits needed to represent  the structural information (for instance if an integer field is constrained to 16 possible values,  say from 0 to 15, PER will represent it on 4 bits, while BER will use one full byte ) and it comes in two flavors,
 aligned and unaligned. For aligned PER, certain types are aligned to byte boundaries using padding bits; unaligned PER is the most economical, at the cost of some extra processing requirements. Unaligned PER stands the comparison to a BER-encoded message compressed with an efficient compression algorithm. PER is typically used for protocols that justify such savings, radio protocols where  channel bandwidth is a scarce resource being good examples. Where bandwidth is not an issue, the less resource-hungry BER can be used.
 
 
 ASN.1 is still intensively used today when publishing telecom standards; the message structures used are annexed to the standard in form of ASN.1 code. Any implementer can pluck it directly and compile it; there is a
 lot less chance of misinterpreting the standard, no need to transfer long tables of messages to code etc;  overall it's a brilliant way of describing a protocol, far superior to the practice in other industry segments.
 
 TTCN-3 owes a lot to ASN.1, into which is heavily rooted ; not incidentally, TTCN-3's way of describing structures is very similar to the one used in ASN.1; the structures can be directly mapped and the TTCN-3 standard itself  describes the translation between ASN.1 and TTCN-3 structures;
 thus, it becomes possible to write TTCN-3 message templates directly against ASN.1 type declarations.
 
 As a consequence  most TTCN-3 tools will understand ASN.1 to a degree and so does Titan. Titan has a rather  evolved ASN.1 compiler included, with one deficiency: it does not support PER encoding. The reasons are manifold and historical: for PER-encoding we have found early a solution using a 3rd party tool, and after that we did not see the business justification of  spending hundreds of man-hours worth, especially that only a relatively small amount of protocols use it.
 
 So what is this solution involving a 3rd party tool consists of? Within Ericsson, we had (and still have ) access to an exceptionally performant and also expensive commercial ASN.1 compiler which was capable of all standard encodings,  so we have came up with the following , a little bit convoluted, but perfectly operational workflow:
 
 -first and foremost , we compile the ASN.1 code with this 3rd party compiler, and generate BER and PER codecs
 -in the encoding/sending direction we first encode with Titan's BER encoder , then decode it with the external BER decoder and recode it with PER
 -in the decoding/receiving direction, first we use the external PER decoder to extract the information into an internal  structure and then we recode in BER and send it to Titan; Titan decodes the BER and regenerates the  structure sent from the other end
 
 Anyone willing to use this workflow had to have both Titan and the external ASN.1 compiler and had to be able to compile the ASN.1 code with both. Due to the commercial nature of the ASN.1 compiler , the same workflow cannot be used in open source projects. To replicate it in open source, we'd need an open source ASN.1 compiler that supports PER encoding.
 
 Luckily, there is such an open source compiler, namely asn1c by Lev Walkin (http://lionet.info/asn1c/blog/  - not to be confused with asn1c from Objective Systems).  Problem is, it comes with its' own restrictions, it does not support Information Object Classes for instance, so it cannot be used for all protocols.  Nonetheless, if it can compile your ASN.1 file, it can be used to generate the PER-codec to be used with Titan.
 
 If you are now sufficiently confused by the fact that three compilers are needed to run a protocol test, let's quickly recap:
 Titan adds the TTCN-3 compiler and an ASN.1 compiler that have in common  the same internal representation, so they can talk to each other directly.  Any external compiler will use its' own internal representation, incompatible with Titan. This external compiler will talk to Titan using the API represented by the BER-codec and it's only role is to deal with PER.
 
 In conclusion, asn1c will work for a category of ASN.1 files and it will not work for others. To find out to which category your ASN.1 code belongs,
 install asn1c ( version 0.9.27 being the latest to date) and try to compile your code with it. If it compiles, you're in.
 
 The exact implementation details can be followed using the attached example of CAM protocol which is part of the ITS communication stack.
 (see ETSI source repository http://forge.etsi.org/websvn/).
 ITS (Intelligent Transport System) is a good example where PER is justifiably used: the cars are connected over wireless  (a wired connection between moving cars has yet to be invented
  ) 
 
 CAM_EncDec.cc is the central part of the code where the BER and PER codecs are invoked according to the logic described above.
 The bunch of .c and .h files have been generated by asn1c ( which creates separate files for each type it encounters)
 with the following command:
 
 
 asn1c gen-PER CAM.asn ITS_Container.asn
 where  -gen-PER  triggers asn1c to generate PER support code.
 
 (Note that we renamed ITS-Container.asn to ITS_Container.asn.  This is an internal Titan convention, all ASN.1 file names containing hyphen will have to be changed accordingly.)
 
 
 Mind that asn1c generates code not entirely compatible with g++. For a full compatibility, some changes are necessarry.
 This might be different from case to case, but for reference here are the changes we have made:
 
 
 
 -Some enum uses have to be prefixed with its' namespace as shown below (APC_CONSTRAINED, APC_UNCONSTRAINED):
 
 
 static asn_per_constraints_t asn_PER_type_AccelerationConfidence_constr_1 GCC_NOTUSED = {
                { asn_per_constraint_t::APC_CONSTRAINED,     7,  7,  0,  102 }    /* (0..102) */,
                { asn_per_constraint_t::APC_UNCONSTRAINED,              -1, -1,  0,  0 },
                0, 0         /* No PER value map */
};
 -Bitwise operators with enums have to be casted to int, and then back to the enumtype:
 
 For example:
 
 Original code:
 
 static asn_per_constraints_t asn_PER_type_PositioningSolutionType_constr_1 GCC_NOTUSED = {
                { APC_CONSTRAINED | APC_EXTENSIBLE,  3,  3,  0,  5 }    /* (0..5,...) */,
                { APC_UNCONSTRAINED,             -1, -1,  0,  0 },
                0, 0         /* No PER value map */
};
 Casted code:
 
 static asn_per_constraints_t asn_PER_type_PositioningSolutionType_constr_1 GCC_NOTUSED = {
                { (asn_per_constraint_t::asn_per_constraint_flags)((int)(asn_per_constraint_t::APC_CONSTRAINED) | (int)(asn_per_constraint_t::APC_EXTENSIBLE)),  3,  3,  0,  5 }        /* (0..5,...) */,
                { asn_per_constraint_t::APC_UNCONSTRAINED,              -1, -1,  0,  0 },
                0, 0         /* No PER value map */
};
 -Specifically cast pointers into the correct type:
 
 
 enc_dyn_arg *arg = (enc_dyn_arg*)key;
uper_ugot_key *arg = (uper_ugot_key*)pd->refill_key;
 -Cast char* variables to const char *:
 
 
 Original code:
static char *_bit_pattern[16] = {
                "0000", "0001", "0010", "0011", "0100", "0101", "0110", "0111",
                "1000", "1001", "1010", "1011", "1100", "1101", "1110", "1111"
};
Casted code:
static const char *_bit_pattern[16] = {
                "0000", "0001", "0010", "0011", "0100", "0101", "0110", "0111",
                "1000", "1001", "1010", "1011", "1100", "1101", "1110", "1111"
};
 
 
 
 To install the example, unpack it , then:
 
 
 cd bin
../install.script
make
./CAM
The execution is a loop encoding-decoding of a CAM message, not very spectacular in itself , but demonstrating the usage of the PER codec.
 As you can see from the log, the three compilers are happily interacting and crunching the unaligned bits of PER-encoding.
 
 In summary, using the example above, anyone can hack a PER codec for his/her  favorite protocol ; the only condition is that asn1c should  support it.
 
 
 
 Regards
 
 
 |  |  |  |  |  |  | 
| Re: Eclipse Titan and the dilemma of PER-encoding [message #1769213 is a reply to message #1763482] | Thu, 27 July 2017 16:02   |  | 
| Eclipse User  |  |  |  |  | Dear Elemer, 
 I have followed your instructions and modified the code as follow:
 
 original code, that I have commented:
 /*
 static asn_per_constraints_t asn_PER_type_AccelerationConfidence_constr_1 GCC_NOTUSED = {
 { APC_CONSTRAINED,	 7,  7,  0,  102 }	// (0..102) ,
 { APC_UNCONSTRAINED,	-1, -1,  0,  0 },
 0, 0	// No PER value map
 };*/
 
 new code that I have added:
 
 static asn_per_constraints_t asn_PER_type_AccelerationConfidence_constr_1 GCC_NOTUSED = {
 { asn_per_constraint_t::APC_CONSTRAINED,     7,  7,  0,  102 }    /* (0..102) */,
 { asn_per_constraint_t::APC_UNCONSTRAINED,              -1, -1,  0,  0 },
 0, 0         /* No PER value map */
 };
 
 
 But I get following Error, can you please let me know how can I fix this.
 
 AccelerationControl.c:123:4: error: 'APC_UNCONSTRAINED' was not declared in this scope
 { APC_UNCONSTRAINED, -1, -1,  0,  0 },
 ^
 AccelerationControl.c:124:4: error: 'APC_CONSTRAINED' was not declared in this scope
 { APC_CONSTRAINED,  0,  0,  7,  7 } /* (SIZE(7..7)) */,
 ^
 make: *** [AccelerationControl.o] Error 1
 
 thank you
 taqi
 |  |  |  |  |  | 
 
 
 Current Time: Sun Oct 26 10:10:57 EDT 2025 
 Powered by FUDForum . Page generated in 0.06575 seconds |