diff options
author | Mohsin Kazmi <sykazmi@cisco.com> | 2018-09-13 09:59:50 +0200 |
---|---|---|
committer | Damjan Marion <dmarion@me.com> | 2018-09-26 15:50:15 +0000 |
commit | a33db2528485ac39e8987bec7b138dbdad21dc5a (patch) | |
tree | 927909d3dcfc914ebd35496f5635b5859b23b83e /extras/japi/java | |
parent | 7b2017ddf7211140e03d9a7329fffc601cdacd6d (diff) |
japi: Move Java API binding to cmake
Change-Id: Ia7ddd9f9e2e2d9c133967593c3e187fe42501eac
Signed-off-by: Mohsin Kazmi <sykazmi@cisco.com>
Diffstat (limited to 'extras/japi/java')
-rw-r--r-- | extras/japi/java/CMakeLists.txt | 176 | ||||
-rw-r--r-- | extras/japi/java/Makefile.am | 279 |
2 files changed, 176 insertions, 279 deletions
diff --git a/extras/japi/java/CMakeLists.txt b/extras/japi/java/CMakeLists.txt new file mode 100644 index 00000000000..f36fac2d4e3 --- /dev/null +++ b/extras/japi/java/CMakeLists.txt @@ -0,0 +1,176 @@ +# Copyright (c) 2018 Cisco and/or its affiliates. +# Licensed under the Apache License, Version 2.0 (the "License"); +# you may not use this file except in compliance with the License. +# You may obtain a copy of the License at: +# +# http://www.apache.org/licenses/LICENSE-2.0 +# +# Unless required by applicable law or agreed to in writing, software +# distributed under the License is distributed on an "AS IS" BASIS, +# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +# See the License for the specific language governing permissions and +# limitations under the License. + +include (UseJava) + +find_path(VNET_INCLUDE_DIR NAMES vnet/api_errno.h) +find_library(VPPINFRA_DIR NAMES vppinfra REQUIRED) +find_library(VLIBMEMORYCLIENT_DIR NAMES vlibmemoryclient REQUIRED) +find_library(SVM_DIR NAMES svm REQUIRED) + +include_directories(${VNET_INCLUDE_DIR} + ${VNET_INCLUDE_DIR}/vpp_plugins + ${JNI_INCLUDE_DIRS} + ${CMAKE_CURRENT_SOURCE_DIR} + ${CMAKE_BINARY_DIR/../vpp/plugins}) +add_compile_options(-Wall) +############# Common package ################## +add_library(jvpp_common SHARED jvpp-common/jvpp_common.c) +set_target_properties(jvpp_common PROPERTIES SOVERSION ${PROJECT_VERSION}) +target_link_libraries(jvpp_common ${VPPINFRA_DIR}) +install(TARGETS jvpp_common DESTINATION lib COMPONENT libjvpp_common) +install(FILES jvpp-common/jvpp_common.h DESTINATION include/japi/) + +set(JVPP_LIBS jvpp_common ${VPPINFRA_DIR} ${VLIBMEMORYCLIENT_DIR} ${SVM_DIR} + Threads::Threads m rt) + +############# Registry package ################## +set(PACKAGE_DIR_JVPP_REGISTRY io/fd/vpp/jvpp) +unset(files) +FILE(GLOB files RELATIVE + ${CMAKE_CURRENT_SOURCE_DIR} + ${CMAKE_CURRENT_SOURCE_DIR}/jvpp-registry/${PACKAGE_DIR_JVPP_REGISTRY}/*.java + ${CMAKE_CURRENT_SOURCE_DIR}/jvpp-registry/${PACKAGE_DIR_JVPP_REGISTRY}/*/*.java +) + +add_custom_target (jvpp-registry-classes) +add_custom_command (TARGET jvpp-registry-classes + PRE_BUILD + COMMAND mkdir -p ${CMAKE_CURRENT_BINARY_DIR}/jvpp-registry + COMMAND ${Java_JAVAC_EXECUTABLE} + ARGS -d ${CMAKE_CURRENT_BINARY_DIR}/jvpp-registry -h jvpp-registry ${files} + WORKING_DIRECTORY ${CMAKE_CURRENT_SOURCE_DIR} +) + +add_library(jvpp_registry SHARED jvpp-registry/jvpp_registry.c) +target_link_libraries(jvpp_registry ${JVPP_LIBS}) +include_directories(jvpp-registry) +add_dependencies(jvpp_registry jvpp_common jvpp-registry-classes) +add_jar(jvpp-registry ${files} VERSION ${PROJECT_VERSION}) +install_jar(jvpp-registry DESTINATION share/java/) + +############## Functions ######################### +function(japigen name) + if(NOT VPP_JAVA_APIGEN) + set(VPP_JAVA_APIGEN ${CMAKE_CURRENT_SOURCE_DIR}/jvpp/gen/jvpp_gen.py) + endif() + add_custom_target(japigen-${name} DEPENDS jvpp-registry) + add_custom_command(TARGET japigen-${name} + POST_BUILD + COMMAND mkdir -p jvpp-${name}/target + COMMAND mkdir -p ${CMAKE_CURRENT_BINARY_DIR}/jvpp-${name} + COMMAND ${VPP_JAVA_APIGEN} + ARGS --plugin_name ${name} --root_dir jvpp-${name} -i ${ARGN} + COMMAND find jvpp-${name} -name \*.java > jvpp-${name}/jvpp-${name}.files + COMMAND ${Java_JAVAC_EXECUTABLE} + ARGS -cp ${CMAKE_CURRENT_BINARY_DIR}/jvpp-registry-${PROJECT_VERSION}.jar -d + ${CMAKE_CURRENT_BINARY_DIR}/jvpp-${name} -h jvpp-${name} + @jvpp-${name}/jvpp-${name}.files + WORKING_DIRECTORY ${CMAKE_CURRENT_SOURCE_DIR} + COMMENT "JAVA_API_GEN ${name}" + ) +endfunction() + +function(jargen name) + add_custom_command(TARGET jvpp_${name} + POST_BUILD + COMMAND cp ${CMAKE_BINARY_DIR}/lib/libjvpp_${name}.so jvpp-${name}/target + COMMAND ${Java_JAR_EXECUTABLE} ARGS cf + ${CMAKE_CURRENT_BINARY_DIR}/jvpp-${name}-${PROJECT_VERSION}.jar + @jvpp-${name}/jvpp-${name}.files + COMMAND rm ARGS -rf jvpp-${name}/target jvpp-${name}/jvpp-${name}.files + jvpp-${name}/jvpp_${name}_gen.h jvpp-${name}/io_fd_vpp_jvpp_*.h + jvpp-registry/io_fd_vpp_jvpp_*.h + WORKING_DIRECTORY ${CMAKE_CURRENT_SOURCE_DIR} + COMMENT "JAR_GEN ${name}" + ) + install( + FILES ${CMAKE_CURRENT_BINARY_DIR}/jvpp-${name}-${PROJECT_VERSION}.jar + DESTINATION share/java + ) +endfunction() + +function(java_api_binding name src_file) + japigen (${name} ${ARGN}) + add_library(jvpp_${name} SHARED jvpp-${name}/jvpp_${src_file}.c) + target_link_libraries(jvpp_${name} ${JVPP_LIBS}) + include_directories(jvpp-${name}) + add_dependencies(jvpp_${name} jvpp_common jvpp_registry japigen-${name}) + jargen (${name}) +endfunction() + +############ Core Package ####################### +unset (files) +unset (JSON_API_PATH) +set (JSON_API_PATH ${CMAKE_BINARY_DIR}/../vpp) + +FILE(GLOB_RECURSE files RELATIVE + ${CMAKE_CURRENT_SOURCE_DIR} + ${JSON_API_PATH}/vnet/*.api.json + ${JSON_API_PATH}/vpp/*.api.json +) + +java_api_binding (core core ${files}) + +############ Plugin Packages ####################### +unset (ACL_JSON_FILE) +unset (NAT_JSON_FILE) +unset (NSH_JSON_FILE) +unset (GTPU_JSON_FILE) +unset (PPPOE_JSON_FILE) +unset (IOAM_TRACE_JSON_FILE) +unset (IOAM_POT_JSON_FILE) +unset (IOAM_EXPORT_JSON_FILE) + +set (plugin_path ${CMAKE_BINARY_DIR}/../vpp/plugins) + +find_file(ACL_JSON_FILE NAMES acl.api.json HINTS ${plugin_path} PATH_SUFFIXES acl) +find_file(NAT_JSON_FILE NAMES nat.api.json HINTS ${plugin_path} PATH_SUFFIXES nat) +find_file(NSH_JSON_FILE NAMES nsh.api.json HINTS ${plugin_path} PATH_SUFFIXES nsh) +find_file(GTPU_JSON_FILE NAMES gtpu.api.json HINTS ${plugin_path} PATH_SUFFIXES gtpu) +find_file(PPPOE_JSON_FILE NAMES pppoe.api.json HINTS ${plugin_path} PATH_SUFFIXES pppoe) +find_file(IOAM_TRACE_JSON_FILE NAMES trace.api.json HINTS ${plugin_path} PATH_SUFFIXES ioam/lib-trace) +find_file(IOAM_POT_JSON_FILE NAMES pot.api.json HINTS ${plugin_path} PATH_SUFFIXES ioam/lib-pot) +find_file(IOAM_EXPORT_JSON_FILE NAMES ioam_export.api.json HINTS ${plugin_path} PATH_SUFFIXES ioam/export) + +if(ACL_JSON_FILE) + java_api_binding (acl acl ${ACL_JSON_FILE}) +endif() + +if(NAT_JSON_FILE) + java_api_binding (nat nat ${NAT_JSON_FILE}) +endif() + +if(NSH_JSON_FILE) + java_api_binding (nsh nsh ${NSH_JSON_FILE}) +endif() + +if(GTPU_JSON_FILE) + java_api_binding (gtpu gtpu ${GTPU_JSON_FILE}) +endif() + +if(PPPOE_JSON_FILE) + java_api_binding (pppoe pppoe ${PPPOE_JSON_FILE}) +endif() + +if(IOAM_TRACE_JSON_FILE) + java_api_binding (ioamtrace ioam_trace ${IOAM_TRACE_JSON_FILE}) +endif() + +if(IOAM_POT_JSON_FILE) + java_api_binding (ioampot ioam_pot ${IOAM_POT_JSON_FILE}) +endif() + +if(IOAM_EXPORT_JSON_FILE) + java_api_binding (ioamexport ioam_export ${IOAM_EXPORT_JSON_FILE}) +endif() diff --git a/extras/japi/java/Makefile.am b/extras/japi/java/Makefile.am deleted file mode 100644 index d33ae82ff0d..00000000000 --- a/extras/japi/java/Makefile.am +++ /dev/null @@ -1,279 +0,0 @@ -# Copyright (c) 2016 Cisco and/or its affiliates. -# Licensed under the Apache License, Version 2.0 (the "License"); -# you may not use this file except in compliance with the License. -# You may obtain a copy of the License at: -# -# http://www.apache.org/licenses/LICENSE-2.0 -# -# Unless required by applicable law or agreed to in writing, software -# distributed under the License is distributed on an "AS IS" BASIS, -# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. -# See the License for the specific language governing permissions and -# limitations under the License. - -AUTOMAKE_OPTIONS = foreign subdir-objects -ACLOCAL_AMFLAGS = -I m4 -AM_LIBTOOLFLAGS = --quiet - -AM_CFLAGS = @CPU_FLAGS@ -Wall -I${top_srcdir} -I${top_builddir} \ - -I$(JAVA_HOME)/include -I$(JAVA_HOME)/include/linux \ - -I@top_srcdir@/plugins -I@top_builddir@/plugins - -AM_LDFLAGS = -shared -avoid-version -rpath /none -no-undefined - -BUILT_SOURCES = -bin_PROGRAMS = -noinst_LTLIBRARIES = -JAR_FILES = -CLEANDIRS = - -# -# jvpp-common -# - -nobase_include_HEADERS = \ - jvpp-common/jvpp_common.h - -lib_LTLIBRARIES = libjvpp_common.la -libjvpp_common_la_SOURCES = jvpp-common/jvpp_common.c -libjvpp_common_la_LDFLAGS = shared -rpath /none -no-undefined -lvppinfra - -JVPP_LIBS = \ - libjvpp_common.la \ - -lvppinfra -lvlibmemoryclient -lsvm -lpthread -lm -lrt - -# -# jvpp-registry (connection management + plugin registry) -# - -noinst_LTLIBRARIES += libjvpp_registry.la - -libjvpp_registry_la_SOURCES = jvpp-registry/jvpp_registry.c -libjvpp_registry_la_CPPFLAGS = -Ijvpp-registry -libjvpp_registry_la_LIBADD = $(JVPP_LIBS) -libjvpp_registry_la_DEPENDENCIES = libjvpp_common.la - -packagedir_jvpp_registry = io/fd/vpp/jvpp -jvpp_registry_src_files := \ - $(wildcard @srcdir@/jvpp-registry/$(packagedir_jvpp_registry)/*.java) \ - $(wildcard @srcdir@/jvpp-registry/$(packagedir_jvpp_registry)/**/*.java) - -BUILT_SOURCES += jvpp-registry/io_fd_vpp_jvpp_VppJNIConnection.h -CLEANDIRS += jvpp-registry/target -JAR_FILES += jvpp-registry-$(PACKAGE_VERSION).jar - -jvpp_registry_ok = jvpp-registry/io_fd_vpp_jvpp_VppJNIConnection.h - -jvpp-registry/io_fd_vpp_jvpp_VppJNIConnection.h: $(jvpp_registry_src_files) - @echo " JVPP GEN $@" - @rm -rf jvpp-registry/target - @mkdir -p jvpp-registry/target - @$(JAVAC) -source 8 -target 8 -d jvpp-registry/target -h jvpp-registry $^ - @touch jvpp-registry.ok - -define japigen - @echo " JVPP GEN $@" - @rm -rf jvpp-$(1)/target - @ @srcdir@/jvpp/gen/jvpp_gen.py --plugin_name $(1) --root_dir jvpp-$(1) \ - -i $(jvpp_$(1)_json_files) - @find jvpp-$(1)/target -name \*.java > jvpp-$(1).generated.files - @find @srcdir@/jvpp-$(1) -name \*.java > jvpp-$(1).static.files - @$(JAVAC) -source 8 -target 8 -classpath jvpp-registry/target \ - -d jvpp-$(1)/target -h jvpp-$(1) @jvpp-$(1).generated.files @jvpp-$(1).static.files -endef - -# -# jvpp-core (Java wrapper for vpe.api) -# -noinst_LTLIBRARIES += libjvpp_core.la -libjvpp_core_la_SOURCES = jvpp-core/jvpp_core.c jvpp-core/jvpp_core_gen.h -libjvpp_core_la_CPPFLAGS = -Ijvpp-registry -Ijvpp-core -libjvpp_core_la_LIBADD = $(JVPP_LIBS) -libjvpp_core_la_DEPENDENCIES = libjvpp_common.la - -BUILT_SOURCES += jvpp-core/io_fd_vpp_jvpp_core_JVppCoreImpl.h -JAR_FILES += jvpp-core-$(PACKAGE_VERSION).jar -CLEANDIRS += jvpp-core/target - -# Include all VPP API files -jvpp_core_json_files = $(shell find @top_builddir@/../vpp/vnet -type f -name '*.api.json') -jvpp_core_json_files += $(shell find @top_builddir@/../vpp/vpp -type f -name '*.api.json') -# except for plugins and unsupported ones: -# jvpp_core_json_files += @top_builddir@/vlibmemory/memclnt.api.json - -jvpp-core/io_fd_vpp_jvpp_core_JVppCoreImpl.h: $(jvpp_registry_ok) $(jvpp_core_json_files) - $(call japigen,core,JVppCoreImpl) - -# -# ACL Plugin -# -if ENABLE_ACL_PLUGIN -noinst_LTLIBRARIES += libjvpp_acl.la -libjvpp_acl_la_SOURCES = jvpp-acl/jvpp_acl.c -libjvpp_acl_la_CPPFLAGS = -Ijvpp-acl -libjvpp_acl_la_LIBADD = $(JVPP_LIBS) -libjvpp_acl_la_DEPENDENCIES = libjvpp_common.la - -BUILT_SOURCES += jvpp-acl/io_fd_vpp_jvpp_acl_JVppAclImpl.h -JAR_FILES += jvpp-acl-$(PACKAGE_VERSION).jar -CLEANDIRS += jvpp-acl/target - -jvpp_acl_json_files = @top_builddir@/../vpp/plugins/acl/acl.api.json - -jvpp-acl/io_fd_vpp_jvpp_acl_JVppAclImpl.h: $(jvpp_registry_ok) $(jvpp_acl_json_files) - $(call japigen,acl,JVppAclImpl) -endif - -# -# GTPU Plugin -# -if ENABLE_GTPU_PLUGIN -noinst_LTLIBRARIES += libjvpp_gtpu.la -libjvpp_gtpu_la_SOURCES = jvpp-gtpu/jvpp_gtpu.c -libjvpp_gtpu_la_CPPFLAGS = -Ijvpp-gtpu -libjvpp_gtpu_la_LIBADD = $(JVPP_LIBS) -libjvpp_gtpu_la_DEPENDENCIES = libjvpp_common.la - -BUILT_SOURCES += jvpp-gtpu/io_fd_vpp_jvpp_gtpu_JVppGtpuImpl.h -JAR_FILES += jvpp-gtpu-$(PACKAGE_VERSION).jar -CLEANDIRS += jvpp-gtpu/target - -jvpp_gtpu_json_files = @top_builddir@/../vpp/plugins/gtpu/gtpu.api.json - -jvpp-gtpu/io_fd_vpp_jvpp_gtpu_JVppGtpuImpl.h: $(jvpp_registry_ok) $(jvpp_gtpu_json_files) - $(call japigen,gtpu,JVppGtpuImpl) -endif - -# -# PPPOE Plugin -# -if ENABLE_PPPOE_PLUGIN -noinst_LTLIBRARIES += libjvpp_pppoe.la -libjvpp_pppoe_la_SOURCES = jvpp-pppoe/jvpp_pppoe.c -libjvpp_pppoe_la_CPPFLAGS = -Ijvpp-pppoe -libjvpp_pppoe_la_LIBADD = $(JVPP_LIBS) -libjvpp_pppoe_la_DEPENDENCIES = libjvpp_common.la - -BUILT_SOURCES += jvpp-pppoe/io_fd_vpp_jvpp_pppoe_JVppPppoeImpl.h -JAR_FILES += jvpp-pppoe-$(PACKAGE_VERSION).jar -CLEANDIRS += jvpp-pppoe/target - -jvpp_pppoe_json_files = @top_builddir@/../vpp/plugins/pppoe/pppoe.api.json - -jvpp-pppoe/io_fd_vpp_jvpp_pppoe_JVppPppoeImpl.h: $(jvpp_registry_ok) $(jvpp_pppoe_json_files) - $(call japigen,pppoe,JVppPppoeImpl) -endif - -# -# NAT Plugin -# -if ENABLE_NAT_PLUGIN -noinst_LTLIBRARIES += libjvpp_nat.la -libjvpp_nat_la_SOURCES = jvpp-nat/jvpp_nat.c -libjvpp_nat_la_CPPFLAGS = -Ijvpp-nat -libjvpp_nat_la_LIBADD = $(JVPP_LIBS) -libjvpp_nat_la_DEPENDENCIES = libjvpp_common.la - -BUILT_SOURCES += jvpp-nat/io_fd_vpp_jvpp_nat_JVppNatImpl.h -JAR_FILES += jvpp-nat-$(PACKAGE_VERSION).jar -CLEANDIRS += jvpp-nat/target - -jvpp_nat_json_files = @top_builddir@/../vpp/plugins/nat/nat.api.json - -jvpp-nat/io_fd_vpp_jvpp_nat_JVppNatImpl.h: $(jvpp_registry_ok) $(jvpp_nat_json_files) - $(call japigen,nat,JVppNatImpl) -endif - -# -# NSH Plugin -# -if ENABLE_NSH_PLUGIN -noinst_LTLIBRARIES += libjvpp_nsh.la -libjvpp_nsh_la_SOURCES = jvpp-nsh/jvpp_nsh.c -libjvpp_nsh_la_CPPFLAGS = -Ijvpp-nsh -libjvpp_nsh_la_LIBADD = $(JVPP_LIBS) -libjvpp_nsh_la_DEPENDENCIES = libjvpp_common.la - -BUILT_SOURCES += jvpp-nsh/io_fd_vpp_jvpp_nsh_JVppNshImpl.h -JAR_FILES += jvpp-nsh-$(PACKAGE_VERSION).jar -CLEANDIRS += jvpp-nsh/target - -jvpp_nsh_json_files = @top_builddir@/../vpp/plugins/nsh/nsh.api.json - -jvpp-nsh/io_fd_vpp_jvpp_nsh_JVppNshImpl.h: $(jvpp_registry_ok) $(jvpp_nsh_json_files) - $(call japigen,nsh,JVppNshImpl) -endif - -# -# iOAM Trace Plugin -# -if ENABLE_IOAM_PLUGIN -noinst_LTLIBRARIES += libjvpp_ioamtrace.la -libjvpp_ioamtrace_la_SOURCES = jvpp-ioamtrace/jvpp_ioam_trace.c -libjvpp_ioamtrace_la_LIBADD = $(JVPP_LIBS) -libjvpp_ioamtrace_la_DEPENDENCIES = libjvpp_common.la - -BUILT_SOURCES += jvpp-ioamtrace/io_fd_vpp_jvpp_ioamtrace_JVppIoamtraceImpl.h -JAR_FILES += jvpp-ioamtrace-$(PACKAGE_VERSION).jar -CLEANDIRS += jvpp-ioamtrace/target - -jvpp_ioamtrace_json_files = @top_builddir@/../vpp/plugins/ioam/lib-trace/trace.api.json - -jvpp-ioamtrace/io_fd_vpp_jvpp_ioamtrace_JVppIoamtraceImpl.h: $(jvpp_registry_ok) $(jvpp_ioamtrace_json_files) - $(call japigen,ioamtrace,JVppIoamtraceImpl) - -# -# iOAM POT Plugin -# -noinst_LTLIBRARIES += libjvpp_ioampot.la -libjvpp_ioampot_la_SOURCES = jvpp-ioampot/jvpp_ioam_pot.c -libjvpp_ioampot_la_LIBADD = $(JVPP_LIBS) -libjvpp_ioampot_la_DEPENDENCIES = libjvpp_common.la - -BUILT_SOURCES += jvpp-ioampot/io_fd_vpp_jvpp_ioampot_JVppIoampotImpl.h -JAR_FILES += jvpp-ioampot-$(PACKAGE_VERSION).jar -CLEANDIRS += jvpp-ioampot/target - -jvpp_ioampot_json_files = @top_builddir@/../vpp/plugins/ioam/lib-pot/pot.api.json - -jvpp-ioampot/io_fd_vpp_jvpp_ioampot_JVppIoampotImpl.h: $(jvpp_registry_ok) $(jvpp_ioampot_json_files) - $(call japigen,ioampot,JVppIoampotImpl) - -# -# iOAM Export Plugin -# -noinst_LTLIBRARIES += libjvpp_ioamexport.la -libjvpp_ioamexport_la_SOURCES = jvpp-ioamexport/jvpp_ioam_export.c -libjvpp_ioamexport_la_LIBADD = $(JVPP_LIBS) -libjvpp_ioamexport_la_DEPENDENCIES = libjvpp_common.la - -BUILT_SOURCES += jvpp-ioamexport/io_fd_vpp_jvpp_ioamexport_JVppIoamexportImpl.h -JAR_FILES += jvpp-ioamexport-$(PACKAGE_VERSION).jar -CLEANDIRS += jvpp-ioamexport/target - -jvpp_ioamexport_json_files = @top_builddir@/../vpp/plugins/ioam/export/ioam_export.api.json - -jvpp-ioamexport/io_fd_vpp_jvpp_ioamexport_JVppIoamexportImpl.h: $(jvpp_registry_ok) $(jvpp_ioamexport_json_files) - $(call japigen,ioamexport,JVppIoamexportImpl) -endif - -# -# JAR creation -# -jvpp-%-$(PACKAGE_VERSION).jar: libjvpp_%.la - @echo " JAR $@" - @cp .libs/libjvpp_$*.so jvpp-$*/target - @$(JAR) cf $(JARFLAGS) $@ -C jvpp-$*/target . - -jardir = $(prefix)/share/java -jar_DATA = $(JAR_FILES) - -all-local: $(JAR_FILES) - -# -# Cleanup -# -CLEANFILES = jvpp-registry.ok $(JAR_FILES) $(BUILT_SOURCES) *.files */*.h - -clean-local: - rm -rf $(CLEANDIRS) |