sclp_info.c 1.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657
  1. /*
  2. * drivers/s390/char/sclp_info.c
  3. *
  4. * Copyright IBM Corp. 2007
  5. * Author(s): Heiko Carstens <heiko.carstens@de.ibm.com>
  6. */
  7. #include <linux/init.h>
  8. #include <linux/errno.h>
  9. #include <linux/string.h>
  10. #include <asm/sclp.h>
  11. #include "sclp.h"
  12. struct sclp_readinfo_sccb s390_readinfo_sccb;
  13. void __init sclp_readinfo_early(void)
  14. {
  15. sclp_cmdw_t command;
  16. struct sccb_header *sccb;
  17. int ret;
  18. __ctl_set_bit(0, 9); /* enable service signal subclass mask */
  19. sccb = &s390_readinfo_sccb.header;
  20. command = SCLP_CMDW_READ_SCP_INFO_FORCED;
  21. while (1) {
  22. u16 response;
  23. memset(&s390_readinfo_sccb, 0, sizeof(s390_readinfo_sccb));
  24. sccb->length = sizeof(s390_readinfo_sccb);
  25. sccb->control_mask[2] = 0x80;
  26. ret = sclp_service_call(command, &s390_readinfo_sccb);
  27. if (ret == -EIO)
  28. goto out;
  29. if (ret == -EBUSY)
  30. continue;
  31. __load_psw_mask(PSW_BASE_BITS | PSW_MASK_EXT |
  32. PSW_MASK_WAIT | PSW_DEFAULT_KEY);
  33. local_irq_disable();
  34. barrier();
  35. response = sccb->response_code;
  36. if (response == 0x10)
  37. break;
  38. if (response != 0x1f0 || command == SCLP_CMDW_READ_SCP_INFO)
  39. break;
  40. command = SCLP_CMDW_READ_SCP_INFO;
  41. }
  42. out:
  43. __ctl_clear_bit(0, 9); /* disable service signal subclass mask */
  44. }