diff options
-rw-r--r-- | net/ceph/osdmap.c | 139 |
1 files changed, 67 insertions, 72 deletions
diff --git a/net/ceph/osdmap.c b/net/ceph/osdmap.c index a350286fd826..6497322d2e3c 100644 --- a/net/ceph/osdmap.c +++ b/net/ceph/osdmap.c | |||
@@ -732,6 +732,67 @@ static int decode_new_pools(void **p, void *end, struct ceph_osdmap *map) | |||
732 | return __decode_pools(p, end, map, true); | 732 | return __decode_pools(p, end, map, true); |
733 | } | 733 | } |
734 | 734 | ||
735 | static int __decode_pg_temp(void **p, void *end, struct ceph_osdmap *map, | ||
736 | bool incremental) | ||
737 | { | ||
738 | u32 n; | ||
739 | |||
740 | ceph_decode_32_safe(p, end, n, e_inval); | ||
741 | while (n--) { | ||
742 | struct ceph_pg pgid; | ||
743 | u32 len, i; | ||
744 | int ret; | ||
745 | |||
746 | ret = ceph_decode_pgid(p, end, &pgid); | ||
747 | if (ret) | ||
748 | return ret; | ||
749 | |||
750 | ceph_decode_32_safe(p, end, len, e_inval); | ||
751 | |||
752 | ret = __remove_pg_mapping(&map->pg_temp, pgid); | ||
753 | BUG_ON(!incremental && ret != -ENOENT); | ||
754 | |||
755 | if (!incremental || len > 0) { | ||
756 | struct ceph_pg_mapping *pg; | ||
757 | |||
758 | ceph_decode_need(p, end, len*sizeof(u32), e_inval); | ||
759 | |||
760 | if (len > (UINT_MAX - sizeof(*pg)) / sizeof(u32)) | ||
761 | return -EINVAL; | ||
762 | |||
763 | pg = kzalloc(sizeof(*pg) + len*sizeof(u32), GFP_NOFS); | ||
764 | if (!pg) | ||
765 | return -ENOMEM; | ||
766 | |||
767 | pg->pgid = pgid; | ||
768 | pg->len = len; | ||
769 | for (i = 0; i < len; i++) | ||
770 | pg->osds[i] = ceph_decode_32(p); | ||
771 | |||
772 | ret = __insert_pg_mapping(pg, &map->pg_temp); | ||
773 | if (ret) { | ||
774 | kfree(pg); | ||
775 | return ret; | ||
776 | } | ||
777 | } | ||
778 | } | ||
779 | |||
780 | return 0; | ||
781 | |||
782 | e_inval: | ||
783 | return -EINVAL; | ||
784 | } | ||
785 | |||
786 | static int decode_pg_temp(void **p, void *end, struct ceph_osdmap *map) | ||
787 | { | ||
788 | return __decode_pg_temp(p, end, map, false); | ||
789 | } | ||
790 | |||
791 | static int decode_new_pg_temp(void **p, void *end, struct ceph_osdmap *map) | ||
792 | { | ||
793 | return __decode_pg_temp(p, end, map, true); | ||
794 | } | ||
795 | |||
735 | /* | 796 | /* |
736 | * decode a full map. | 797 | * decode a full map. |
737 | */ | 798 | */ |
@@ -810,36 +871,9 @@ static int osdmap_decode(void **p, void *end, struct ceph_osdmap *map) | |||
810 | ceph_decode_addr(&map->osd_addr[i]); | 871 | ceph_decode_addr(&map->osd_addr[i]); |
811 | 872 | ||
812 | /* pg_temp */ | 873 | /* pg_temp */ |
813 | ceph_decode_32_safe(p, end, len, e_inval); | 874 | err = decode_pg_temp(p, end, map); |
814 | for (i = 0; i < len; i++) { | 875 | if (err) |
815 | int n, j; | 876 | goto bad; |
816 | struct ceph_pg pgid; | ||
817 | struct ceph_pg_mapping *pg; | ||
818 | |||
819 | err = ceph_decode_pgid(p, end, &pgid); | ||
820 | if (err) | ||
821 | goto bad; | ||
822 | ceph_decode_need(p, end, sizeof(u32), e_inval); | ||
823 | n = ceph_decode_32(p); | ||
824 | if (n > (UINT_MAX - sizeof(*pg)) / sizeof(u32)) | ||
825 | goto e_inval; | ||
826 | ceph_decode_need(p, end, n * sizeof(u32), e_inval); | ||
827 | pg = kmalloc(sizeof(*pg) + n*sizeof(u32), GFP_NOFS); | ||
828 | if (!pg) { | ||
829 | err = -ENOMEM; | ||
830 | goto bad; | ||
831 | } | ||
832 | pg->pgid = pgid; | ||
833 | pg->len = n; | ||
834 | for (j = 0; j < n; j++) | ||
835 | pg->osds[j] = ceph_decode_32(p); | ||
836 | |||
837 | err = __insert_pg_mapping(pg, &map->pg_temp); | ||
838 | if (err) | ||
839 | goto bad; | ||
840 | dout(" added pg_temp %lld.%x len %d\n", pgid.pool, pgid.seed, | ||
841 | len); | ||
842 | } | ||
843 | 877 | ||
844 | /* crush */ | 878 | /* crush */ |
845 | ceph_decode_32_safe(p, end, len, e_inval); | 879 | ceph_decode_32_safe(p, end, len, e_inval); |
@@ -1038,48 +1072,9 @@ struct ceph_osdmap *osdmap_apply_incremental(void **p, void *end, | |||
1038 | } | 1072 | } |
1039 | 1073 | ||
1040 | /* new_pg_temp */ | 1074 | /* new_pg_temp */ |
1041 | ceph_decode_32_safe(p, end, len, e_inval); | 1075 | err = decode_new_pg_temp(p, end, map); |
1042 | while (len--) { | 1076 | if (err) |
1043 | struct ceph_pg_mapping *pg; | 1077 | goto bad; |
1044 | int j; | ||
1045 | struct ceph_pg pgid; | ||
1046 | u32 pglen; | ||
1047 | |||
1048 | err = ceph_decode_pgid(p, end, &pgid); | ||
1049 | if (err) | ||
1050 | goto bad; | ||
1051 | ceph_decode_need(p, end, sizeof(u32), e_inval); | ||
1052 | pglen = ceph_decode_32(p); | ||
1053 | if (pglen) { | ||
1054 | ceph_decode_need(p, end, pglen*sizeof(u32), e_inval); | ||
1055 | |||
1056 | /* removing existing (if any) */ | ||
1057 | (void) __remove_pg_mapping(&map->pg_temp, pgid); | ||
1058 | |||
1059 | /* insert */ | ||
1060 | if (pglen > (UINT_MAX - sizeof(*pg)) / sizeof(u32)) | ||
1061 | goto e_inval; | ||
1062 | pg = kmalloc(sizeof(*pg) + sizeof(u32)*pglen, GFP_NOFS); | ||
1063 | if (!pg) { | ||
1064 | err = -ENOMEM; | ||
1065 | goto bad; | ||
1066 | } | ||
1067 | pg->pgid = pgid; | ||
1068 | pg->len = pglen; | ||
1069 | for (j = 0; j < pglen; j++) | ||
1070 | pg->osds[j] = ceph_decode_32(p); | ||
1071 | err = __insert_pg_mapping(pg, &map->pg_temp); | ||
1072 | if (err) { | ||
1073 | kfree(pg); | ||
1074 | goto bad; | ||
1075 | } | ||
1076 | dout(" added pg_temp %lld.%x len %d\n", pgid.pool, | ||
1077 | pgid.seed, pglen); | ||
1078 | } else { | ||
1079 | /* remove */ | ||
1080 | __remove_pg_mapping(&map->pg_temp, pgid); | ||
1081 | } | ||
1082 | } | ||
1083 | 1078 | ||
1084 | /* ignore the rest */ | 1079 | /* ignore the rest */ |
1085 | *p = end; | 1080 | *p = end; |