Re: [PATCH] MM: Support more pagesizes for MAP_HUGETLB/SHM_HUGETLB v4

2012-10-13 Thread Andi Kleen
> > -   struct user_struct **user, int creat_flags);
> > +   struct user_struct **user, int creat_flags,
> > +   int page_size_log);
> > +int hugetlb_get_quota(struct address_space *mapping, long delta);
> > +void hugetlb_put_quota(struct address_space *mapping, long delta);
> > +
> > +int hugetlb_get_quota(struct address_space *mapping, long delta);
> > +void hugetlb_put_quota(struct address_space *mapping, long delta);
> 
> 
> For what to add(twice) hugetlb_get/put_quota?

Hmm probably a merge error. Thanks for catching. 

I can repost or post the trivial incremential.

-Andi

-- 
a...@linux.intel.com -- Speaking for myself only.
--
To unsubscribe from this list: send the line "unsubscribe linux-kernel" in
the body of a message to majord...@vger.kernel.org
More majordomo info at  http://vger.kernel.org/majordomo-info.html
Please read the FAQ at  http://www.tux.org/lkml/


Re: [PATCH] MM: Support more pagesizes for MAP_HUGETLB/SHM_HUGETLB v4

2012-10-13 Thread Hillf Danton
Hi Andi,

On Fri, Oct 12, 2012 at 7:53 AM, Andi Kleen  wrote:
> diff --git a/include/linux/hugetlb.h b/include/linux/hugetlb.h
> index 2251648..c626a2a 100644
> --- a/include/linux/hugetlb.h
> +++ b/include/linux/hugetlb.h
> @@ -183,7 +183,13 @@ extern const struct file_operations 
> hugetlbfs_file_operations;
>  extern const struct vm_operations_struct hugetlb_vm_ops;
>  struct file *hugetlb_file_setup(const char *name, unsigned long addr,
> size_t size, vm_flags_t acct,
> -   struct user_struct **user, int creat_flags);
> +   struct user_struct **user, int creat_flags,
> +   int page_size_log);
> +int hugetlb_get_quota(struct address_space *mapping, long delta);
> +void hugetlb_put_quota(struct address_space *mapping, long delta);
> +
> +int hugetlb_get_quota(struct address_space *mapping, long delta);
> +void hugetlb_put_quota(struct address_space *mapping, long delta);


For what to add(twice) hugetlb_get/put_quota?

Hillf
--
To unsubscribe from this list: send the line "unsubscribe linux-kernel" in
the body of a message to majord...@vger.kernel.org
More majordomo info at  http://vger.kernel.org/majordomo-info.html
Please read the FAQ at  http://www.tux.org/lkml/


Re: [PATCH] MM: Support more pagesizes for MAP_HUGETLB/SHM_HUGETLB v4

2012-10-13 Thread Hillf Danton
Hi Andi,

On Fri, Oct 12, 2012 at 7:53 AM, Andi Kleen a...@firstfloor.org wrote:
 diff --git a/include/linux/hugetlb.h b/include/linux/hugetlb.h
 index 2251648..c626a2a 100644
 --- a/include/linux/hugetlb.h
 +++ b/include/linux/hugetlb.h
 @@ -183,7 +183,13 @@ extern const struct file_operations 
 hugetlbfs_file_operations;
  extern const struct vm_operations_struct hugetlb_vm_ops;
  struct file *hugetlb_file_setup(const char *name, unsigned long addr,
 size_t size, vm_flags_t acct,
 -   struct user_struct **user, int creat_flags);
 +   struct user_struct **user, int creat_flags,
 +   int page_size_log);
 +int hugetlb_get_quota(struct address_space *mapping, long delta);
 +void hugetlb_put_quota(struct address_space *mapping, long delta);
 +
 +int hugetlb_get_quota(struct address_space *mapping, long delta);
 +void hugetlb_put_quota(struct address_space *mapping, long delta);


For what to add(twice) hugetlb_get/put_quota?

Hillf
--
To unsubscribe from this list: send the line unsubscribe linux-kernel in
the body of a message to majord...@vger.kernel.org
More majordomo info at  http://vger.kernel.org/majordomo-info.html
Please read the FAQ at  http://www.tux.org/lkml/


Re: [PATCH] MM: Support more pagesizes for MAP_HUGETLB/SHM_HUGETLB v4

2012-10-13 Thread Andi Kleen
  -   struct user_struct **user, int creat_flags);
  +   struct user_struct **user, int creat_flags,
  +   int page_size_log);
  +int hugetlb_get_quota(struct address_space *mapping, long delta);
  +void hugetlb_put_quota(struct address_space *mapping, long delta);
  +
  +int hugetlb_get_quota(struct address_space *mapping, long delta);
  +void hugetlb_put_quota(struct address_space *mapping, long delta);
 
 
 For what to add(twice) hugetlb_get/put_quota?

Hmm probably a merge error. Thanks for catching. 

I can repost or post the trivial incremential.

-Andi

-- 
a...@linux.intel.com -- Speaking for myself only.
--
To unsubscribe from this list: send the line unsubscribe linux-kernel in
the body of a message to majord...@vger.kernel.org
More majordomo info at  http://vger.kernel.org/majordomo-info.html
Please read the FAQ at  http://www.tux.org/lkml/


[PATCH] MM: Support more pagesizes for MAP_HUGETLB/SHM_HUGETLB v4

2012-10-11 Thread Andi Kleen
From: Andi Kleen 

There was some desire in large applications using MAP_HUGETLB/SHM_HUGETLB
to use 1GB huge pages on some mappings, and stay with 2MB on others. This
is useful together with NUMA policy: use 2MB interleaving on some mappings,
but 1GB on local mappings.

This patch extends the IPC/SHM syscall interfaces slightly to allow specifying
the page size.

It borrows some upper bits in the existing flag arguments and allows encoding
the log of the desired page size in addition to the *_HUGETLB flag.
When 0 is specified the default size is used, this makes the change fully
compatible.

Extending the internal hugetlb code to handle this is straight forward. Instead
of a single mount it just keeps an array of them and selects the right
mount based on the specified page size. When no page size is specified
it uses the mount of the default page size.

The change is not visible in /proc/mounts because internal mounts
don't appear there. It also has very little overhead: the additional
mounts just consume a super block, but not more memory when not used.

I also exported the new flags to the user headers
(they were previously under __KERNEL__). Right now only symbols
for x86 and some other architecture for 1GB and 2MB are defined.
The interface should already work for all other architectures
though.  Only architectures that define multiple hugetlb sizes
actually need it (that is currently x86, tile, powerpc). However
tile and powerpc have user configurable hugetlb sizes, so it's
not easy to add defines. A program on those architectures would
need to query sysfs and use the appropiate log2.

v2: Port to new tree. Fix unmount.
v3: Ported to latest tree.
v4: Ported to latest tree. Minor changes for review feedback. Updated
description.
Acked-by: Rik van Riel 
Acked-by: KAMEZAWA Hiroyuki 
Signed-off-by: Andi Kleen 
---
 arch/x86/include/asm/mman.h |3 ++
 fs/hugetlbfs/inode.c|   63 +++
 include/linux/hugetlb.h |   12 ++-
 include/linux/shm.h |   19 
 include/uapi/asm-generic/mman.h |   13 
 ipc/shm.c   |3 +-
 mm/mmap.c   |5 ++-
 7 files changed, 100 insertions(+), 18 deletions(-)

diff --git a/arch/x86/include/asm/mman.h b/arch/x86/include/asm/mman.h
index 593e51d..513b05f 100644
--- a/arch/x86/include/asm/mman.h
+++ b/arch/x86/include/asm/mman.h
@@ -3,6 +3,9 @@
 
 #define MAP_32BIT  0x40/* only give out 32bit addresses */
 
+#define MAP_HUGE_2MB(21 << MAP_HUGE_SHIFT)
+#define MAP_HUGE_1GB(30 << MAP_HUGE_SHIFT)
+
 #include 
 
 #endif /* _ASM_X86_MMAN_H */
diff --git a/fs/hugetlbfs/inode.c b/fs/hugetlbfs/inode.c
index c5bc355..d34bb56 100644
--- a/fs/hugetlbfs/inode.c
+++ b/fs/hugetlbfs/inode.c
@@ -923,7 +923,7 @@ static struct file_system_type hugetlbfs_fs_type = {
.kill_sb= kill_litter_super,
 };
 
-static struct vfsmount *hugetlbfs_vfsmount;
+static struct vfsmount *hugetlbfs_vfsmount[HUGE_MAX_HSTATE];
 
 static int can_do_hugetlb_shm(void)
 {
@@ -932,9 +932,22 @@ static int can_do_hugetlb_shm(void)
return capable(CAP_IPC_LOCK) || in_group_p(shm_group);
 }
 
+static int get_hstate_idx(int page_size_log)
+{
+   struct hstate *h;
+
+   if (!page_size_log)
+   return default_hstate_idx;
+   h = size_to_hstate(1 << page_size_log);
+   if (!h)
+   return -1;
+   return h - hstates;
+}
+
 struct file *hugetlb_file_setup(const char *name, unsigned long addr,
size_t size, vm_flags_t acctflag,
-   struct user_struct **user, int creat_flags)
+   struct user_struct **user,
+   int creat_flags, int page_size_log)
 {
int error = -ENOMEM;
struct file *file;
@@ -944,9 +957,14 @@ struct file *hugetlb_file_setup(const char *name, unsigned 
long addr,
struct qstr quick_string;
struct hstate *hstate;
unsigned long num_pages;
+   int hstate_idx;
+
+   hstate_idx = get_hstate_idx(page_size_log);
+   if (hstate_idx < 0)
+   return ERR_PTR(-ENODEV);
 
*user = NULL;
-   if (!hugetlbfs_vfsmount)
+   if (!hugetlbfs_vfsmount[hstate_idx])
return ERR_PTR(-ENOENT);
 
if (creat_flags == HUGETLB_SHMFS_INODE && !can_do_hugetlb_shm()) {
@@ -963,7 +981,7 @@ struct file *hugetlb_file_setup(const char *name, unsigned 
long addr,
}
}
 
-   root = hugetlbfs_vfsmount->mnt_root;
+   root = hugetlbfs_vfsmount[hstate_idx]->mnt_root;
quick_string.name = name;
quick_string.len = strlen(quick_string.name);
quick_string.hash = 0;
@@ -971,7 +989,7 @@ struct file *hugetlb_file_setup(const char *name, unsigned 
long addr,
if (!path.dentry)
goto out_shm_unlock;
 
-   path.mnt = mntget(hugetlbfs_vfsmount);
+   path.mnt = 

[PATCH] MM: Support more pagesizes for MAP_HUGETLB/SHM_HUGETLB v4

2012-10-11 Thread Andi Kleen
From: Andi Kleen a...@linux.intel.com

There was some desire in large applications using MAP_HUGETLB/SHM_HUGETLB
to use 1GB huge pages on some mappings, and stay with 2MB on others. This
is useful together with NUMA policy: use 2MB interleaving on some mappings,
but 1GB on local mappings.

This patch extends the IPC/SHM syscall interfaces slightly to allow specifying
the page size.

It borrows some upper bits in the existing flag arguments and allows encoding
the log of the desired page size in addition to the *_HUGETLB flag.
When 0 is specified the default size is used, this makes the change fully
compatible.

Extending the internal hugetlb code to handle this is straight forward. Instead
of a single mount it just keeps an array of them and selects the right
mount based on the specified page size. When no page size is specified
it uses the mount of the default page size.

The change is not visible in /proc/mounts because internal mounts
don't appear there. It also has very little overhead: the additional
mounts just consume a super block, but not more memory when not used.

I also exported the new flags to the user headers
(they were previously under __KERNEL__). Right now only symbols
for x86 and some other architecture for 1GB and 2MB are defined.
The interface should already work for all other architectures
though.  Only architectures that define multiple hugetlb sizes
actually need it (that is currently x86, tile, powerpc). However
tile and powerpc have user configurable hugetlb sizes, so it's
not easy to add defines. A program on those architectures would
need to query sysfs and use the appropiate log2.

v2: Port to new tree. Fix unmount.
v3: Ported to latest tree.
v4: Ported to latest tree. Minor changes for review feedback. Updated
description.
Acked-by: Rik van Riel r...@redhat.com
Acked-by: KAMEZAWA Hiroyuki kamezawa.hir...@jp.fujitsu.com
Signed-off-by: Andi Kleen a...@linux.intel.com
---
 arch/x86/include/asm/mman.h |3 ++
 fs/hugetlbfs/inode.c|   63 +++
 include/linux/hugetlb.h |   12 ++-
 include/linux/shm.h |   19 
 include/uapi/asm-generic/mman.h |   13 
 ipc/shm.c   |3 +-
 mm/mmap.c   |5 ++-
 7 files changed, 100 insertions(+), 18 deletions(-)

diff --git a/arch/x86/include/asm/mman.h b/arch/x86/include/asm/mman.h
index 593e51d..513b05f 100644
--- a/arch/x86/include/asm/mman.h
+++ b/arch/x86/include/asm/mman.h
@@ -3,6 +3,9 @@
 
 #define MAP_32BIT  0x40/* only give out 32bit addresses */
 
+#define MAP_HUGE_2MB(21  MAP_HUGE_SHIFT)
+#define MAP_HUGE_1GB(30  MAP_HUGE_SHIFT)
+
 #include asm-generic/mman.h
 
 #endif /* _ASM_X86_MMAN_H */
diff --git a/fs/hugetlbfs/inode.c b/fs/hugetlbfs/inode.c
index c5bc355..d34bb56 100644
--- a/fs/hugetlbfs/inode.c
+++ b/fs/hugetlbfs/inode.c
@@ -923,7 +923,7 @@ static struct file_system_type hugetlbfs_fs_type = {
.kill_sb= kill_litter_super,
 };
 
-static struct vfsmount *hugetlbfs_vfsmount;
+static struct vfsmount *hugetlbfs_vfsmount[HUGE_MAX_HSTATE];
 
 static int can_do_hugetlb_shm(void)
 {
@@ -932,9 +932,22 @@ static int can_do_hugetlb_shm(void)
return capable(CAP_IPC_LOCK) || in_group_p(shm_group);
 }
 
+static int get_hstate_idx(int page_size_log)
+{
+   struct hstate *h;
+
+   if (!page_size_log)
+   return default_hstate_idx;
+   h = size_to_hstate(1  page_size_log);
+   if (!h)
+   return -1;
+   return h - hstates;
+}
+
 struct file *hugetlb_file_setup(const char *name, unsigned long addr,
size_t size, vm_flags_t acctflag,
-   struct user_struct **user, int creat_flags)
+   struct user_struct **user,
+   int creat_flags, int page_size_log)
 {
int error = -ENOMEM;
struct file *file;
@@ -944,9 +957,14 @@ struct file *hugetlb_file_setup(const char *name, unsigned 
long addr,
struct qstr quick_string;
struct hstate *hstate;
unsigned long num_pages;
+   int hstate_idx;
+
+   hstate_idx = get_hstate_idx(page_size_log);
+   if (hstate_idx  0)
+   return ERR_PTR(-ENODEV);
 
*user = NULL;
-   if (!hugetlbfs_vfsmount)
+   if (!hugetlbfs_vfsmount[hstate_idx])
return ERR_PTR(-ENOENT);
 
if (creat_flags == HUGETLB_SHMFS_INODE  !can_do_hugetlb_shm()) {
@@ -963,7 +981,7 @@ struct file *hugetlb_file_setup(const char *name, unsigned 
long addr,
}
}
 
-   root = hugetlbfs_vfsmount-mnt_root;
+   root = hugetlbfs_vfsmount[hstate_idx]-mnt_root;
quick_string.name = name;
quick_string.len = strlen(quick_string.name);
quick_string.hash = 0;
@@ -971,7 +989,7 @@ struct file *hugetlb_file_setup(const char *name, unsigned 
long addr,
if (!path.dentry)