I'm running into issues with the MSP430 and use of PSImode in packed
structures.  Here the native word size is HImode, but all general purpose
registers are capable of holding and operating on PSImode (=20 bit) values.

Due to instruction set limitations, on this architecture:

  (subreg:HI (reg:PSI 0))

is fine, but

  (subreg:HI (reg:PSI 2))

is inexpressible.  The upper 4 bits can only be accessed with multiple shift
instructions.

CANNOT_CHANGE_MODE_CLASS doesn't provide the control to express this
restriction on subregs.  The only way I've found to tell gcc this is to hack
validate_subreg to rule out non-zero offsets on PSImode.

Is there a non-hack way to tell GCC about this limitation?

More generally, is there a way to tell GCC not to try to decompose SImode
operations into subregs even though word_mode is HImode?  I've got
HARD_REGNO_NREGS, HARD_REGNO_NREGS_HAS_PADDING, and REGMODE_NATURAL_SIZE set
up as accurately as I can:

#define MSP430_MODE_NREGS(MODE)                                         \
  (PSImode == (MODE)                                                    \
   ? 1                                                                  \
   : ((GET_MODE_SIZE (MODE) + UNITS_PER_WORD - 1) / UNITS_PER_WORD))
#define HARD_REGNO_NREGS(REGNO, MODE) MSP430_MODE_NREGS(MODE)
#define HARD_REGNO_MODE_OK(REGNO, MODE) 1
#define MODES_TIEABLE_P(MODE1, MODE2)                           \
  ((PSImode != (MODE1) || PSImode == (MODE2))                   \
   && (PSImode != (MODE2) || 1 == MSP430_MODE_NREGS (MODE1)))
#define HARD_REGNO_NREGS_HAS_PADDING(REGNO, MODE) (PSImode == (MODE))
#define HARD_REGNO_NREGS_WITH_PADDING(REGNO, MODE) (PSImode == (MODE)
? 2 : MSP430_MODE_NREGS(MODE))
#define REGMODE_NATURAL_SIZE(MODE) (PSImode == (MODE) ?
2*UNITS_PER_WORD : UNITS_PER_WORD)

Thanks.

Peter

Reply via email to