+ if (asm_flags & ASM_CONSTRAINT_FLAG_INVALID) {
+ errorf(&statement->base.source_position,
+ "some constraints in '%s' are invalid", constraints);
+ continue;
+ }
+ if (asm_flags & ASM_CONSTRAINT_FLAG_MODIFIER_WRITE) {
+ errorf(&statement->base.source_position,
+ "write flag specified for input constraints '%s'",
+ constraints);
+ continue;
+ }
+
+ ir_node *input;
+ if ( (asm_flags & ASM_CONSTRAINT_FLAG_SUPPORTS_IMMEDIATE)
+ || (asm_flags & ASM_CONSTRAINT_FLAG_SUPPORTS_REGISTER) ) {
+ /* we can treat this as "normal" input */
+ input = expression_to_firm(argument->expression);
+ } else if (asm_flags & ASM_CONSTRAINT_FLAG_SUPPORTS_MEMOP) {
+ /* pure memory ops need no input (but we have to make sure we
+ * attach to the memory) */
+ assert(! (asm_flags &
+ (ASM_CONSTRAINT_FLAG_SUPPORTS_IMMEDIATE
+ | ASM_CONSTRAINT_FLAG_SUPPORTS_REGISTER)));
+ needs_memory = true;
+ input = expression_to_addr(argument->expression);
+ } else {
+ errorf(&statement->base.source_position,
+ "only modifiers but no place set in constraints '%s'",
+ constraints);
+ continue;
+ }
+
+ ir_asm_constraint constraint;
+ constraint.pos = next_pos++;
+ constraint.constraint = new_id_from_str(constraints);
+ constraint.mode = get_irn_mode(input);
+
+ obstack_grow(&asm_obst, &constraint, sizeof(constraint));
+ ins[in_size++] = input;
+ }
+
+ if (needs_memory) {
+ ir_asm_constraint constraint;
+ constraint.pos = next_pos++;
+ constraint.constraint = new_id_from_str("");
+ constraint.mode = mode_M;
+
+ obstack_grow(&asm_obst, &constraint, sizeof(constraint));
+ ins[in_size++] = get_store();
+ }
+
+ assert(obstack_object_size(&asm_obst)
+ == in_size * sizeof(ir_asm_constraint));
+ ir_asm_constraint *input_constraints = obstack_finish(&asm_obst);
+
+ /* create asm node */
+ dbg_info *dbgi = get_dbg_info(&statement->base.source_position);
+
+ ident *asm_text = new_id_from_str(statement->asm_text.begin);
+
+ ir_node *node = new_d_ASM(dbgi, in_size, ins, input_constraints,
+ out_size, output_constraints,
+ n_clobbers, clobbers, asm_text);
+
+ if (statement->is_volatile) {
+ set_irn_pinned(node, op_pin_state_pinned);
+ } else {
+ set_irn_pinned(node, op_pin_state_floats);
+ }
+
+ /* create output projs & connect them */
+ if (needs_memory) {
+ ir_node *projm = new_Proj(node, mode_M, out_size+1);
+ set_store(projm);
+ }
+
+ size_t i;
+ for (i = 0; i < out_size; ++i) {
+ const expression_t *out_expr = out_exprs[i];
+ long pn = i;
+ ir_mode *mode = get_ir_mode(out_expr->base.type);
+ ir_node *proj = new_Proj(node, mode, pn);
+ ir_node *addr = out_addrs[i];
+
+ set_value_for_expression_addr(out_expr, proj, addr);